mirror of
				https://github.com/e107inc/e107.git
				synced 2025-10-25 11:46:49 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			9194 lines
		
	
	
		
			214 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			9194 lines
		
	
	
		
			214 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
| <?php
 | |
| /*
 | |
|  * e107 website system
 | |
|  *
 | |
|  * Copyright (C) 2008-2012 e107 Inc (e107.org)
 | |
|  * Released under the terms and conditions of the
 | |
|  * GNU General Public License (http://www.gnu.org/licenses/gpl.txt)
 | |
|  *
 | |
|  * Administration User Interface logic
 | |
|  *
 | |
|  * $URL$
 | |
|  * $Id$
 | |
|  */
 | |
| 
 | |
|  /**
 | |
|  * @package e107
 | |
|  * @subpackage e107_handlers
 | |
|  * @version $Id$
 | |
|  *
 | |
|  * Administration User Interface logic
 | |
|  */
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @todo core request handler (non-admin), core response
 | |
|  */
 | |
| if (!defined('e107_INIT')){ exit; }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  */
 | |
| class e_admin_request
 | |
| {
 | |
| 	/**
 | |
| 	 * Current GET request array
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $_request_qry;
 | |
| 
 | |
| 	/**
 | |
| 	 * Current POST array
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $_posted_qry;
 | |
| 
 | |
| 	/**
 | |
| 	 * Current Mode
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_mode = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Default Mode
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_default_mode = 'main';
 | |
| 
 | |
| 	/**
 | |
| 	 * Key name for mode search
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_mode_key = 'mode';
 | |
| 
 | |
| 	/**
 | |
| 	 * Current action
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_action = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Default Action
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_default_action = 'index';
 | |
| 
 | |
| 	/**
 | |
| 	 * Key name for action search
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_action_key = 'action';
 | |
| 
 | |
| 	/**
 | |
| 	 * Current ID
 | |
| 	 * @var integer
 | |
| 	 */
 | |
| 	protected $_id = 0;
 | |
| 
 | |
| 	/**
 | |
| 	 * Key name for ID search
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_id_key = 'id';
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor
 | |
| 	 *
 | |
| 	 * @param string|null $request_string [optional]
 | |
| 	 * @param bool $parse
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function __construct($request_string = null, $parse = true)
 | |
| 	{
 | |
| 		if($request_string === null)
 | |
| 		{
 | |
| 			$request_string = str_replace('&', '&', e_QUERY);
 | |
| 		}
 | |
| 		if($parse)
 | |
| 		{
 | |
| 			$this->parseRequest($request_string);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Parse request data
 | |
| 	 * @param string|array $request_data
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	protected function parseRequest($request_data)
 | |
| 	{
 | |
| 		if(is_string($request_data))
 | |
| 		{
 | |
| 			parse_str($request_data, $request_data);
 | |
| 		}
 | |
| 		$this->_request_qry = (array) $request_data;
 | |
| 
 | |
| 		// Set current mode
 | |
| 		if(isset($this->_request_qry[$this->_mode_key]))
 | |
| 		{
 | |
| 			$this->_mode = preg_replace('/[\W]/', '', $this->_request_qry[$this->_mode_key]);
 | |
| 		}
 | |
| 
 | |
| 		// Set current action
 | |
| 		if(isset($this->_request_qry[$this->_action_key]))
 | |
| 		{
 | |
| 			$this->_action = preg_replace('/[\W]/', '', $this->_request_qry[$this->_action_key]);
 | |
| 		}
 | |
| 
 | |
| 		// Set current id
 | |
| 		if(isset($this->_request_qry[$this->_id_key]))
 | |
| 		{
 | |
| 			$this->_id = preg_replace('/[^\w\-:\.]/', '', $this->_request_qry[$this->_id_key]);
 | |
| 		}
 | |
| 
 | |
| 		$this->_posted_qry =& $_POST; //raw?
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieve variable from GET scope
 | |
| 	 * If $key is null, all GET data will be returned
 | |
| 	 *
 | |
| 	 * @param string $key [optional]
 | |
| 	 * @param mixed $default [optional]
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function getQuery($key = null, $default = null)
 | |
| 	{
 | |
| 		if($key === null)
 | |
| 		{
 | |
| 			return $this->_request_qry;
 | |
| 		}
 | |
| 		return (isset($this->_request_qry[$key]) ? $this->_request_qry[$key] : $default);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set/Unset GET variable
 | |
| 	 * If $key is array, $value is not used.
 | |
| 	 * If $value is null, (string) $key is unset
 | |
| 	 *
 | |
| 	 * @param string|array $key
 | |
| 	 * @param mixed $value [optional]
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setQuery($key, $value = null)
 | |
| 	{
 | |
| 		if(is_array($key))
 | |
| 		{
 | |
| 			foreach ($key as $k=>$v)
 | |
| 			{
 | |
| 				$this->setQuery($k, $v);
 | |
| 			}
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		if($value === null)
 | |
| 		{
 | |
| 			unset($this->_request_qry[$key], $_GET[$key]);
 | |
| 
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		$this->_request_qry[$key] = $value;
 | |
| 		$_GET[$key] = $value;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieve variable from POST scope
 | |
| 	 * If $key is null, all POST data will be returned
 | |
| 	 *
 | |
| 	 * @param string $key [optional]
 | |
| 	 * @param mixed $default [optional]
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function getPosted($key = null, $default = null)
 | |
| 	{
 | |
| 		if($key === null)
 | |
| 		{
 | |
| 			return $this->_posted_qry;
 | |
| 		}
 | |
| 		return (isset($this->_posted_qry[$key]) ? $this->_posted_qry[$key] : $default);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set/Unset POST variable
 | |
| 	 * If $key is array, $value is not used.
 | |
| 	 * If $value is null, (string) $key is unset
 | |
| 	 *
 | |
| 	 * @param string|array $key
 | |
| 	 * @param object $value [optional]
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setPosted($key, $value = null)
 | |
| 	{
 | |
| 		if(is_array($key))
 | |
| 		{
 | |
| 			if(empty($key))
 | |
| 			{
 | |
| 				$this->_posted_qry = array(); //POST reset
 | |
| 				return $this;
 | |
| 			}
 | |
| 			foreach ($key as $k=>$v)
 | |
| 			{
 | |
| 				$this->setPosted($k, $v);
 | |
| 			}
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		if($value === null)
 | |
| 		{
 | |
| 			unset($this->_posted_qry[$key]);
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		$tp = e107::getParser();
 | |
| 		$this->_posted_qry[$tp->post_toForm($key)] = $tp->post_toForm($value);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current mode
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getMode()
 | |
| 	{
 | |
| 		if(!$this->_mode)
 | |
| 		{
 | |
| 			return $this->getDefaultMode();
 | |
| 		}
 | |
| 		return $this->_mode;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get default mode
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getDefaultMode()
 | |
| 	{
 | |
| 		return $this->_default_mode;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current mode name
 | |
| 	 *
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getModeName()
 | |
| 	{
 | |
| 		return strtolower(str_replace('-', '_', $this->getMode()));
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reset current mode
 | |
| 	 * @param string $mode
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setMode($mode)
 | |
| 	{
 | |
| 		$this->_mode = preg_replace('/[\W]/', '', $mode);
 | |
| 		$this->setQuery($this->_mode_key, $this->_mode);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set default mode
 | |
| 	 * @param string $mode
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setDefaultMode($mode)
 | |
| 	{
 | |
| 		if($mode)
 | |
| 		{
 | |
| 			$this->_default_mode = $mode;
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set mode key name
 | |
| 	 * @param string $key
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setModeKey($key)
 | |
| 	{
 | |
| 		$this->_mode_key = $key;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current action
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getAction()
 | |
| 	{
 | |
| 		if(!$this->_action)
 | |
| 		{
 | |
| 			return $this->getDefaultAction();
 | |
| 		}
 | |
| 		return $this->_action;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get default action
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getDefaultAction()
 | |
| 	{
 | |
| 		return $this->_default_action;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current action name
 | |
| 	 * @return string camelized action
 | |
| 	 */
 | |
| 	public function getActionName()
 | |
| 	{
 | |
| 		return $this->camelize($this->getAction());
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reset current action
 | |
| 	 *
 | |
| 	 * @param string $action
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setAction($action)
 | |
| 	{
 | |
| 		$this->_action = preg_replace('/[\W]/', '', $action);
 | |
| 		$this->setQuery($this->_action_key, $this->_action);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set default action
 | |
| 	 *
 | |
| 	 * @param string $action
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setDefaultAction($action)
 | |
| 	{
 | |
| 		if($action)
 | |
| 		{
 | |
| 			$this->_default_action = $action;
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set action key name
 | |
| 	 * @param string $key
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setActionKey($key)
 | |
| 	{
 | |
| 		$this->_action_key = $key;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current ID
 | |
| 	 * @return integer
 | |
| 	 */
 | |
| 	public function getId()
 | |
| 	{
 | |
| 		return $this->_id;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reset current ID
 | |
| 	 * @param string $id
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setId($id)
 | |
| 	{
 | |
| 		$id = (int) $id;
 | |
| 		$this->_id = $id;
 | |
| 		$this->setQuery($this->_id_key, $id);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set id key name
 | |
| 	 * @param string $key
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function setIdKey($key)
 | |
| 	{
 | |
| 		$this->_id_key = $key;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Build query string from current request array
 | |
| 	 * NOTE: changing url separator to & ($encode==true) (thus URL XHTML compliance) works in PHP 5.1.2+ environment
 | |
| 	 *
 | |
| 	 * @param string|array $merge_with [optional] override request values
 | |
| 	 * @param boolean $encode if true & separator will be used, all values will be http encoded, default true
 | |
| 	 * @param string|array $exclude_from_query numeric array/comma separated list of vars to be excluded from current query, true - don't use current query at all
 | |
| 	 * @param boolean $keepSpecial don't exclude special vars as 'mode' and 'action'
 | |
| 	 * @return string url encoded query string
 | |
| 	 */
 | |
| 	public function buildQueryString($merge_with = array(), $encode = true, $exclude_from_query = '', $keepSpecial = true)
 | |
| 	{
 | |
| 		$ret = $this->getQuery();
 | |
| 
 | |
| 		//special case - exclude all current
 | |
| 		if($exclude_from_query === true)
 | |
| 		{
 | |
| 			$exclude_from_query = array_keys($ret);
 | |
| 		}
 | |
| 		// to array
 | |
| 		if(is_string($exclude_from_query))
 | |
| 		{
 | |
| 			$exclude_from_query = array_map('trim', explode(',', $exclude_from_query));
 | |
| 		}
 | |
| 		if($exclude_from_query)
 | |
| 		{
 | |
| 			foreach ($exclude_from_query as $var)
 | |
| 			{
 | |
| 				if($keepSpecial && $var != $this->_action_key && $var != $this->_mode_key)
 | |
| 				{
 | |
| 					unset($ret[$var]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if(is_string($merge_with))
 | |
| 		{
 | |
| 			parse_str($merge_with, $merge_with);
 | |
| 		}
 | |
| 		$ret = array_merge($ret, (array) $merge_with);
 | |
| 		$separator = '&';
 | |
| 		if($encode)
 | |
| 		{
 | |
| 			$separator = '&';
 | |
| 			//$ret = array_map('rawurlencode', $ret);
 | |
| 		}
 | |
| 
 | |
| 		$ret = http_build_query($ret, 'numeric_', $separator);
 | |
| 		if(!$encode)
 | |
| 		{
 | |
| 			return rawurldecode($ret);
 | |
| 		}
 | |
| 		return $ret;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Convert string to CamelCase
 | |
| 	 *
 | |
| 	 * @param string $str
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function camelize($str)
 | |
| 	{
 | |
| 		return implode('', array_map('ucfirst', explode('-', str_replace('_', '-', $str))));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * TODO - front response parent, should do all the header.php work
 | |
|  */
 | |
| class e_admin_response
 | |
| {
 | |
| 	/**
 | |
| 	 * Body segments
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $_body = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Title segments
 | |
| 	 *
 | |
| 	 * @var unknown_type
 | |
| 	 */
 | |
| 	protected $_title = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * e107 meta title
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $_e_PAGETITLE = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * e107 meta description
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $_META_DESCRIPTION = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * e107 meta keywords
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $_META_KEYWORDS = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Render mods
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $_render_mod = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Meta title segment description
 | |
| 	 *
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_meta_title_separator = ' - ';
 | |
| 
 | |
| 	/**
 | |
| 	 * Title segment separator
 | |
| 	 *
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $_title_separator = ' » ';
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor
 | |
| 	 *
 | |
| 	 */
 | |
| 	public function __construct()
 | |
| 	{
 | |
| 		$this->_render_mod['default'] = 'admin_page';
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set body segments for a namespace
 | |
| 	 *
 | |
| 	 * @param string $content
 | |
| 	 * @param string $namespace segment namesapce
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function setBody($content, $namespace = 'default')
 | |
| 	{
 | |
| 		$this->_body[$namespace] = $content;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Append body segment to a namespace
 | |
| 	 *
 | |
| 	 * @param string $content
 | |
| 	 * @param string $namespace segment namesapce
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function appendBody($content, $namespace = 'default')
 | |
| 	{
 | |
| 		if(!isset($this->_body[$namespace]))
 | |
| 		{
 | |
| 			$this->_body[$namespace] = array();
 | |
| 		}
 | |
| 		$this->_body[$namespace][] = $content;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Prepend body segment to a namespace
 | |
| 	 *
 | |
| 	 * @param string $content
 | |
| 	 * @param string $namespace segment namespace
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function prependBody($content, $namespace = 'default')
 | |
| 	{
 | |
| 		if(!isset($this->_body[$namespace]))
 | |
| 		{
 | |
| 			$this->_body[$namespace] = array();
 | |
| 		}
 | |
| 		$this->_body[$namespace] = array_merge(array($content), $this->_body[$namespace]);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get body segments from a namespace
 | |
| 	 *
 | |
| 	 * @param string $namespace segment namesapce
 | |
| 	 * @param boolean $reset reset segment namespace
 | |
| 	 * @param string|boolean $glue if false return array, else return string
 | |
| 	 * @return string|array
 | |
| 	 */
 | |
| 	public function getBody($namespace = 'default', $reset = false, $glue = '')
 | |
| 	{
 | |
| 		$content = vartrue($this->_body[$namespace], array());
 | |
| 		if($reset)
 | |
| 		{
 | |
| 			$this->_body[$namespace] = array();
 | |
| 		}
 | |
| 		if(is_bool($glue))
 | |
| 		{
 | |
| 			return ($glue ? $content : implode('', $content));
 | |
| 		}
 | |
| 		return implode($glue, $content);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set title segments for a namespace
 | |
| 	 *
 | |
| 	 * @param string $title
 | |
| 	 * @param string $namespace
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function setTitle($title, $namespace = 'default')
 | |
| 	{
 | |
| 		$this->_title[$namespace] = array($title);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Append title segment to a namespace
 | |
| 	 *
 | |
| 	 * @param string $title
 | |
| 	 * @param string $namespace segment namesapce
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function appendTitle($title, $namespace = 'default')
 | |
| 	{
 | |
| 		if(empty($title))
 | |
| 		{
 | |
| 			return $this;
 | |
| 		}
 | |
| 		if(!isset($this->_title[$namespace]))
 | |
| 		{
 | |
| 			$this->_title[$namespace] = array();
 | |
| 		}
 | |
| 		$this->_title[$namespace][] = $title;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Prepend title segment to a namespace
 | |
| 	 *
 | |
| 	 * @param string $title
 | |
| 	 * @param string $namespace segment namespace
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function prependTitle($title, $namespace = 'default')
 | |
| 	{
 | |
| 		if(empty($title))
 | |
| 		{
 | |
| 			return $this;
 | |
| 		}
 | |
| 		if(!isset($this->_title[$namespace]))
 | |
| 		{
 | |
| 			$this->_title[$namespace] = array();
 | |
| 		}
 | |
| 		$this->_title[$namespace] = array_merge(array($title), $this->_title[$namespace]);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get title segments from namespace
 | |
| 	 *
 | |
| 	 * @param string $namespace
 | |
| 	 * @param boolean $reset
 | |
| 	 * @param boolean|string $glue
 | |
| 	 * @return string|array
 | |
| 	 */
 | |
| 	public function getTitle($namespace = 'default', $reset = false, $glue = '  ')
 | |
| 	{
 | |
| 		$content = array();
 | |
| 				
 | |
| 		if(isset($this->_title[$namespace]) && is_array($this->_title[$namespace]))
 | |
| 		{
 | |
| 			$content = $this->_title[$namespace];
 | |
| 		}
 | |
| 		if($reset)
 | |
| 		{
 | |
| 			unset($this->_title[$namespace]);
 | |
| 		}
 | |
| 		if(is_bool($glue) || empty($glue))
 | |
| 		{
 | |
| 			return ($glue ? $content : implode($this->_title_separator, $content));
 | |
| 		}
 | |
| 
 | |
| 		$glue = deftrue('SEP',' - '); // Defined by admin theme. // admin-ui used only by bootstrap. 
 | |
| 
 | |
| 		return implode($glue, $content);
 | |
| 		// return $head. implode($glue, $content).$foot;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set render mode for a namespace
 | |
| 	 *
 | |
| 	 * @param string $render_mod
 | |
| 	 * @param string $namespace
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function setRenderMod($render_mod, $namespace = 'default')
 | |
| 	{
 | |
| 		$this->_render_mod[$namespace] = $render_mod;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set render mode for namespace
 | |
| 	 *
 | |
| 	 * @param string $namespace
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getRenderMod($namespace = 'default')
 | |
| 	{
 | |
| 		return varset($this->_render_mod[$namespace], null);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add meta title, description and keywords segments
 | |
| 	 *
 | |
| 	 * @param string $meta property name
 | |
| 	 * @param string $content meta content
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function addMetaData($meta, $content)
 | |
| 	{
 | |
| 		$tp = e107::getParser();
 | |
| 		$meta = '_' . $meta;
 | |
| 		if(isset($this->{$meta}) && !empty($content))
 | |
| 		{
 | |
| 			$this->{$meta}[] = strip_tags($content);
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add meta title segment
 | |
| 	 *
 | |
| 	 * @param string $title
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function addMetaTitle($title)
 | |
| 	{
 | |
| 		$this->addMetaData('e_PAGETITLE', $title);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the Meta-title (overrides existing)
 | |
| 	 * @param string $title
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function setMetaTitle($title)
 | |
| 	{
 | |
| 		$meta = '_e_PAGETITLE';
 | |
| 		$this->{$meta} = array(strip_tags($title));
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add meta description segment
 | |
| 	 *
 | |
| 	 * @param string $description
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function addMetaDescription($description)
 | |
| 	{
 | |
| 		$this->addMetaData('META_DESCRIPTION', $description);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add meta keywords segment
 | |
| 	 *
 | |
| 	 * @param string $keyword
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function addMetaKeywords($keyword)
 | |
| 	{
 | |
| 		$this->addMetaData('META_KEYWORDS', $keyword);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Send e107 meta-data
 | |
| 	 *
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function sendMeta()
 | |
| 	{
 | |
| 		//HEADERF already included or meta content already sent
 | |
| 		if(e_AJAX_REQUEST || defined('HEADER_INIT') || defined('e_PAGETITLE'))
 | |
| 		{
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		if(!defined('e_PAGETITLE') && !empty($this->_e_PAGETITLE))
 | |
| 		{
 | |
| 			define('e_PAGETITLE', implode($this->_meta_title_separator, $this->_e_PAGETITLE));
 | |
| 		}
 | |
| 
 | |
| 		if(!defined('META_DESCRIPTION') && !empty($this->_META_DESCRIPTION))
 | |
| 		{
 | |
| 			define('META_DESCRIPTION', implode(' ', $this->_META_DESCRIPTION));
 | |
| 		}
 | |
| 		if(!defined('META_KEYWORDS') && !empty($this->_META_KEYWORDS))
 | |
| 		{
 | |
| 			define('META_KEYWORDS', implode(', ', $this->_META_KEYWORDS));
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add content segment to the header namespace
 | |
| 	 *
 | |
| 	 * @param string $content
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function addHeaderContent($content)
 | |
| 	{
 | |
| 		$this->appendBody($content, 'header_content');
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get page header namespace content segments
 | |
| 	 *
 | |
| 	 * @param boolean $reset
 | |
| 	 * @param boolean $glue
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getHeaderContent($reset = true, $glue = "\n\n")
 | |
| 	{
 | |
| 		return $this->getBody('header_content', $reset, $glue);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Switch to iframe mod
 | |
| 	 * FIXME - implement e_IFRAME to frontend - header_default.php
 | |
| 	 *
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function setIframeMod()
 | |
| 	{
 | |
| 		global $HEADER, $FOOTER, $CUSTOMHEADER, $CUSTOMFOOTER;
 | |
| 		$FOOTER = '';
 | |
| 		$HEADER = $FOOTER;
 | |
| 		$CUSTOMFOOTER = array();
 | |
| 		$CUSTOMHEADER = $CUSTOMFOOTER;
 | |
| 		//TODO generic $_GET to activate for any page of admin. 
 | |
| 		// New
 | |
| 		if(!defined('e_IFRAME'))
 | |
| 		{
 | |
| 			define('e_IFRAME', true);
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Send Response Output
 | |
| 	 *
 | |
| 	 * @param string $name segment
 | |
| 	 * @param array $options valid keys are: messages|render|meta|return|raw|ajax
 | |
| 	 * @return array|string|null
 | |
| 	 */
 | |
| 	public function send($name = 'default', $options = array())
 | |
| 	{
 | |
| 		if(is_string($options))
 | |
| 		{
 | |
| 			parse_str($options, $options);
 | |
| 		}
 | |
| 
 | |
| 		// Merge with all available default options
 | |
| 		$options = array_merge(array(
 | |
| 			'messages' => true,
 | |
| 			'render' => true,
 | |
| 			'meta' => false,
 | |
| 			'return' => false,
 | |
| 			'raw' => false,
 | |
| 			'ajax' => false
 | |
| 		), $options);
 | |
| 
 | |
| 		$content = $this->getBody($name, true);
 | |
| 		$title = $this->getTitle($name, true);
 | |
| 		$return = $options['return'];
 | |
| 
 | |
| 		if($options['ajax'] || e_AJAX_REQUEST)
 | |
| 		{
 | |
| 			$type = $options['ajax'] && is_string($options['ajax']) ? $options['ajax'] : '';
 | |
| 			$this->getJsHelper()->sendResponse($type);
 | |
| 		}
 | |
| 
 | |
| 		if($options['messages'])
 | |
| 		{
 | |
| 			$content = e107::getMessage()->render().$content;
 | |
| 		}
 | |
| 
 | |
| 		if($options['meta'])
 | |
| 		{
 | |
| 			$this->sendMeta();
 | |
| 		}
 | |
| 
 | |
| 		// raw output expected - force return array
 | |
| 		if($options['raw'])
 | |
| 		{
 | |
| 			return array($title, $content, $this->getRenderMod($name));
 | |
| 		}
 | |
| 
 | |
| 		//render disabled by the controller
 | |
| 		if(!$this->getRenderMod($name))
 | |
| 		{
 | |
| 			$options['render'] = false;
 | |
| 		}
 | |
| 
 | |
| 		if($options['render'])
 | |
| 		{
 | |
| 			return e107::getRender()->tablerender($title, $content, $this->getRenderMod($name), $return);
 | |
| 		}
 | |
| 
 | |
| 		if($return)
 | |
| 		{
 | |
| 			return $content;
 | |
| 		}
 | |
| 
 | |
| 		print($content);
 | |
| 		return '';
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get JS Helper instance
 | |
| 	 *
 | |
| 	 * @return e_jshelper
 | |
| 	 */
 | |
| 	public function getJsHelper()
 | |
| 	{
 | |
| 		return e107::getSingleton('e_jshelper', true, 'admin_response');
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * TODO - request related code should be moved to core
 | |
|  * request handler
 | |
|  */
 | |
| class e_admin_dispatcher
 | |
| {
 | |
| 	/**
 | |
| 	 * @var e_admin_request
 | |
| 	 */
 | |
| 	protected $_request;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var e_admin_response
 | |
| 	 */
 | |
| 	protected $_response;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var e_admin_controller
 | |
| 	 */
 | |
| 	protected $_current_controller;
 | |
| 
 | |
| 	/**
 | |
| 	 * Required (set by child class).
 | |
| 	 * Controller map array in format
 | |
| 	 * 'MODE' => array('controller' =>'CONTROLLER_CLASS_NAME'[, 'path' => 'CONTROLLER SCRIPT PATH', 'ui' => extend of 'comments_admin_form_ui', 'uipath' => 'path/to/ui/']);
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $modes = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * Optional - access restrictions per action 
 | |
| 	 * Access array in format (similar to adminMenu)
 | |
| 	 * 'MODE/ACTION' => e_UC_* (userclass constant, or custom userclass ID if dynamically set) 
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $access = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * Optional - generic entry point access restriction (via getperms()) 
 | |
| 	 * Value of this for plugins would be always 'P'.
 | |
| 	 * When an array is detected, route mode/action = admin perms is used. (similar to $access)
 | |
| 	 * More detailed access control is granted with $access and $modes[MODE]['perm'] or  $modes[MODE]['userclass'] settings
 | |
| 	 *
 | |
| 	 * @var string|array
 | |
| 	 */
 | |
| 	protected $perm;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $defaultMode = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $defaultAction = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Optional - map 'mode/action' pair to 'modeAlias/actionAlias'
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $adminMenuAliases = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Optional (set by child class).
 | |
| 	 * Required for admin menu render
 | |
| 	 * Format: 'mode/action' => array('caption' => 'Link title'[, 'perm' => '0', 'url' => '{e_PLUGIN}plugname/admin_config.php'], ...);
 | |
| 	 * Note that 'perm' and 'userclass' restrictions are inherited from the $modes, $access and $perm, so you don't have to set that vars if 
 | |
| 	 * you don't need any additional 'visual' control.
 | |
| 	 * All valid key-value pair (see e107::getNav()->admin function) are accepted.
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $adminMenu = array();
 | |
| 	
 | |
| 
 | |
| 	protected $adminMenuIcon;
 | |
| 	/**
 | |
| 	 * Optional (set by child class).
 | |
| 	 * Page titles for pages not in adminMenu (e.g. main/edit)
 | |
| 	 * Format array(mod/action => Page Title)
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $pageTitles = array(
 | |
| 		'main/edit' => LAN_MANAGE,
 | |
| 	);
 | |
| 
 | |
| 	/**
 | |
| 	 * Optional (set by child class).
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $menuTitle = 'Menu';
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $pluginTitle = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor
 | |
| 	 *
 | |
| 	 * @param string|array|e_admin_request $request [optional]
 | |
| 	 * @param e_admin_response $response
 | |
| 	 */
 | |
| 	public function __construct($auto_observe = true, $request = null, $response = null)
 | |
| 	{
 | |
| 		// we let know some admin routines we are in UI mod - related with some legacy checks and fixes
 | |
| 		if(!defined('e_ADMIN_UI'))
 | |
| 		{
 | |
| 			define('e_ADMIN_UI', true);
 | |
| 		}
 | |
| 
 | |
| 		if(!empty($_GET['iframe']) && !defined('e_IFRAME'))
 | |
| 		{
 | |
| 			define('e_IFRAME', true);
 | |
| 		}
 | |
| 
 | |
| 		require_once(e_ADMIN.'boot.php');
 | |
| 		
 | |
| 		if($request === null || !is_object($request))
 | |
| 		{
 | |
| 			$request = new e_admin_request($request);
 | |
| 		}
 | |
| 
 | |
| 		if($response === null)
 | |
| 		{
 | |
| 			$response = new e_admin_response();
 | |
| 		}
 | |
| 
 | |
| 		$this->setRequest($request)->setResponse($response)->init();
 | |
| 
 | |
| 		if(!$this->defaultMode || !$this->defaultAction)
 | |
| 		{
 | |
| 			$this->setDefaults();
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 		// current user does not have access to default route, so find a new one.
 | |
| 		if(!$hasAccess = $this->hasRouteAccess($this->defaultMode.'/'.$this->defaultAction))
 | |
| 		{
 | |
| 			if($newRoute = $this->getApprovedAccessRoute())
 | |
| 			{
 | |
| 				list($this->defaultMode,$this->defaultAction) = explode('/',$newRoute);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		$request->setDefaultMode($this->defaultMode)->setDefaultAction($this->defaultAction);
 | |
| 
 | |
| 		// register itself
 | |
| 		e107::setRegistry('admin/ui/dispatcher', $this);
 | |
| 		
 | |
| 		// permissions and restrictions
 | |
| 		$this->checkAccess();
 | |
| 
 | |
| 		if($auto_observe)
 | |
| 		{
 | |
| 			$this->runObservers();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined constructor - called before _initController() method
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function init()
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function checkAccess()
 | |
| 	{
 | |
| 		$request = $this->getRequest();
 | |
| 		$currentMode = $request->getMode();
 | |
| 
 | |
| 		// access based on mode setting - general controller access
 | |
| 		if(!$this->hasModeAccess($currentMode))
 | |
| 		{
 | |
| 			$request->setAction('e403');
 | |
| 			e107::getMessage()->addError(LAN_NO_PERMISSIONS)
 | |
| 				->addDebug('Mode access restriction triggered.');
 | |
| 			return false;
 | |
| 		}
 | |
| 		
 | |
| 		// access based on $access settings - access per action
 | |
| 		$currentAction = $request->getAction();
 | |
| 		$route = $currentMode.'/'.$currentAction;
 | |
| 
 | |
| 
 | |
| 
 | |
| 		if(!$this->hasRouteAccess($route))
 | |
| 		{
 | |
| 			$request->setAction('e403');
 | |
| 			e107::getMessage()->addError(LAN_NO_PERMISSIONS)
 | |
| 				->addDebug('Route access restriction triggered:'.$route);
 | |
| 			return false;
 | |
| 		}
 | |
| 		
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $mode
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function hasModeAccess($mode)
 | |
| 	{
 | |
| 		// mode userclass (former check_class())
 | |
| 		if(isset($this->modes[$mode]['userclass']) && !e107::getUser()->checkClass($this->modes[$mode]['userclass'], false))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 		// mode admin permission (former getperms())
 | |
| 		if(isset($this->modes[$mode]['perm']) && !e107::getUser()->checkAdminPerms($this->modes[$mode]['perm']))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 		
 | |
| 		// generic dispatcher admin permission  (former getperms())
 | |
| 		if($this->perm !== null && is_string($this->perm) && !e107::getUser()->checkAdminPerms($this->perm))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $route
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function hasRouteAccess($route)
 | |
| 	{
 | |
| 		if(empty($route))
 | |
| 		{
 | |
| 		    return true;
 | |
| 		}
 | |
| 
 | |
| 		if(isset($this->access[$route]) && !check_class($this->access[$route]))
 | |
| 		{
 | |
| 			e107::getMessage()->addDebug('Userclass Permissions Failed: ' .$this->access[$route]);
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		if(is_array($this->perm) && isset($this->perm[$route]) && !getperms($this->perm[$route]))
 | |
| 		{
 | |
| 			e107::getMessage()->addDebug('Admin Permissions Failed.' .$this->perm[$route]);
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieve missing default action/mode
 | |
| 	 * @return e_admin_dispatcher
 | |
| 	 */
 | |
| 	public function setDefaults()
 | |
| 	{
 | |
| 		// try Admin menu first
 | |
| 		if($this->adminMenu)
 | |
| 		{
 | |
| 			reset($this->adminMenu);
 | |
| 			list($mode, $action) = explode('/', key($this->adminMenu), 3);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			reset($this->modes);
 | |
| 			$mode = key($this->modes);
 | |
| 			$action = $this->modes[$mode]['index'];
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 		if(!$this->defaultMode)
 | |
| 		{
 | |
| 			$this->defaultMode = $mode;
 | |
| 		}
 | |
| 		if(!$this->defaultAction)
 | |
| 		{
 | |
| 			$this->defaultAction = $action;
 | |
| 		}
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Search through access for an approved route.
 | |
| 	 * Returns false if no approved route found.
 | |
| 	 *
 | |
| 	 * @return string|bool
 | |
| 	 */
 | |
| 	private function getApprovedAccessRoute()
 | |
| 	{
 | |
| 		if(empty($this->access))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		foreach($this->access as $route=>$uclass)
 | |
| 		{
 | |
| 			if(check_class($uclass))
 | |
| 			{
 | |
| 				return $route;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get admin menu array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getMenuData()
 | |
| 	{
 | |
| 		return $this->adminMenu;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Sets the administrative menu data.
 | |
| 	 *
 | |
| 	 * @param array $menu The menu data to set.
 | |
| 	 * @return $this
 | |
| 	 */
 | |
| 	public function setMenuData($menu)
 | |
| 	{
 | |
| 		$this->adminMenu = $menu;
 | |
| 		return $this;
 | |
| 	}
 | |
| 	
 | |
| 	/**
 | |
| 	 * Get admin menu array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getPageTitles()
 | |
| 	{
 | |
| 		return $this->pageTitles;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get admin menu array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getMenuAliases()
 | |
| 	{
 | |
| 		return $this->adminMenuAliases;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieves the menu icon associated with the admin panel.
 | |
| 	 *
 | |
| 	 * @return string The menu icon.
 | |
| 	 */
 | |
| 	public function getMenuIcon()
 | |
| 	{
 | |
| 		return $this->adminMenuIcon;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieves the menu title.
 | |
| 	 *
 | |
| 	 * @return string The title of the menu.
 | |
| 	 */
 | |
| 	public function getMenuTitle()
 | |
| 	{
 | |
| 		return $this->menuTitle;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get request object
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function getRequest()
 | |
| 	{
 | |
| 		return $this->_request;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set request object
 | |
| 	 * @param e_admin_request $request
 | |
| 	 * @return e_admin_dispatcher
 | |
| 	 */
 | |
| 	public function setRequest($request)
 | |
| 	{
 | |
| 		$this->_request = $request;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get response object
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function getResponse()
 | |
| 	{
 | |
| 		return $this->_response;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set response object
 | |
| 	 * @param e_admin_response $response
 | |
| 	 * @return e_admin_dispatcher
 | |
| 	 */
 | |
| 	public function setResponse($response)
 | |
| 	{
 | |
| 		$this->_response = $response;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Dispatch & render all
 | |
| 	 *
 | |
| 	 * @param boolean $run_header see runObservers()
 | |
| 	 * @param boolean $return see runPage()
 | |
| 	 * @return string|array current admin page body
 | |
| 	 */
 | |
| 	public function run($run_header = true, $return = 'render')
 | |
| 	{
 | |
| 		return $this->runObservers()->runPage($return);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Run observers/headers only, should be called before header.php call
 | |
| 	 *
 | |
| 	 * @return e_admin_dispatcher
 | |
| 	 */
 | |
| 	public function runObservers($run_header = true)
 | |
| 	{
 | |
| 		//search for $actionName.'Observer' method. Additional $actionName.$triggerName.'Trigger' methods will be called as well
 | |
| 		$this->getController()->dispatchObserver();
 | |
| 
 | |
| 		//search for $actionName.'Header' method, js manager should be used inside for sending JS to the page,
 | |
| 		// meta information should be created there as well
 | |
| 		if($run_header)
 | |
| 		{
 | |
| 			$this->getController()->dispatchHeader();
 | |
| 
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Run page action.
 | |
| 	 * If return type is array, it should contain allowed response options (see e_admin_response::send())
 | |
| 	 * Available return type string values:
 | |
| 	 * - render_return: return rendered content ( see e107::getRender()->tablerender()), add system messages, send meta information
 | |
| 	 * - render: outputs rendered content ( see e107::getRender()->tablerender()), add system messages
 | |
| 	 * - response: return response object
 | |
| 	 * - raw: return array(title, content, render mode)
 | |
| 	 * - ajax: force ajax output (and exit)
 | |
| 	 *
 | |
| 	 * @param string|array $return_type expected string values: render|render_out|response|raw|ajax[_text|_json|_xml]
 | |
| 	 * @return array|e_admin_response|string|null
 | |
| 	 */
 | |
| 	public function runPage($return_type = 'render')
 | |
| 	{
 | |
| 		$response = $this->getController()->dispatchPage();
 | |
| 		if(is_array($return_type))
 | |
| 		{
 | |
| 			return $response->send('default', $return_type);
 | |
| 		}
 | |
| 		switch($return_type)
 | |
| 		{
 | |
| 			case 'render_return':
 | |
| 				$options = array(
 | |
| 					'messages' => true,
 | |
| 					'render' => true,
 | |
| 					'meta' => true,
 | |
| 					'return' => true,
 | |
| 					'raw' => false
 | |
| 				);
 | |
| 			break;
 | |
| 
 | |
| 			case 'raw':
 | |
| 				$options = array(
 | |
| 					'messages' => false,
 | |
| 					'render' => false,
 | |
| 					'meta' => false,
 | |
| 					'return' => true,
 | |
| 					'raw' => true
 | |
| 				);
 | |
| 			break;
 | |
| 
 | |
| 			case 'ajax':
 | |
| 			case 'ajax_text':
 | |
| 			case 'ajax_xml';
 | |
| 			case 'ajax_json';
 | |
| 				$options = array(
 | |
| 					'messages' => false,
 | |
| 					'render' => false,
 | |
| 					'meta' => false,
 | |
| 					'return' => false,
 | |
| 					'raw' => false,
 | |
| 					'ajax' => str_replace(array('ajax_', 'ajax'), array('', 'text'), $return_type)
 | |
| 				);
 | |
| 			break;
 | |
| 
 | |
| 			case 'response':
 | |
| 				return $response;
 | |
| 			break;
 | |
| 
 | |
| 			case 'render':
 | |
| 			default:
 | |
| 				$options = array(
 | |
| 					'messages' => true,
 | |
| 					'render' => true,
 | |
| 					'meta' => false,
 | |
| 					'return' => false,
 | |
| 					'raw' => false
 | |
| 				);
 | |
| 			break;
 | |
| 		}
 | |
| 		return $response->send('default', $options);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Get perms
 | |
| 	 * @return array|string
 | |
| 	 */
 | |
| 	public function getPerm()
 | |
| 	{
 | |
| 		return $this->perm;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Proxy method
 | |
| 	 *
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getHeader()
 | |
| 	{
 | |
| 		return $this->getController()->getHeader();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current controller object
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function getController()
 | |
| 	{
 | |
| 		if($this->_current_controller === null)
 | |
| 		{
 | |
| 			$this->_initController();
 | |
| 		}
 | |
| 		return $this->_current_controller;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Try to init Controller from request using current controller map
 | |
| 	 *
 | |
| 	 * @return e_admin_dispatcher
 | |
| 	 */
 | |
| 	protected function _initController()
 | |
| 	{
 | |
| 		$request = $this->getRequest();
 | |
| 		$response = $this->getResponse();
 | |
| 		if(isset($this->modes[$request->getModeName()]) && isset($this->modes[$request->getModeName()]['controller']))
 | |
| 		{
 | |
| 			$class_name = $this->modes[$request->getModeName()]['controller'];
 | |
| 			$class_path = vartrue($this->modes[$request->getModeName()]['path']);
 | |
| 
 | |
| 			if($class_path)
 | |
| 			{
 | |
| 				require_once(e107::getParser()->replaceConstants($class_path));
 | |
| 			}
 | |
| 			if($class_name && class_exists($class_name))//NOTE: autoload in the play
 | |
| 			{
 | |
| 				$this->_current_controller = new  $class_name($request, $response);
 | |
| 				//give access to current request object, user defined init
 | |
| 				$this->_current_controller->setRequest($this->getRequest())->init();
 | |
| 			}
 | |
| 			// Known controller (found in e_admin_dispatcher::$modes), class not found exception
 | |
| 			else
 | |
| 			{
 | |
| 				// TODO - admin log
 | |
| 				// get default controller
 | |
| 				$this->_current_controller = $this->getDefaultController();
 | |
| 				// add messages
 | |
| 				e107::getMessage()->add('Can\'t find class <strong>"'.($class_name ? $class_name : 'n/a').'"</strong> for controller <strong>"'.ucfirst($request->getModeName()).'"</strong>', E_MESSAGE_ERROR)
 | |
| 					->add('Requested: '.e_REQUEST_SELF.'?'.$request->buildQueryString(), E_MESSAGE_DEBUG);
 | |
| 				//
 | |
| 				$request->setMode($this->getDefaultControllerName())->setAction('e404');
 | |
| 				$this->_current_controller->setRequest($request)->init();
 | |
| 			}
 | |
| 
 | |
| 			if(!empty($this->modes[$request->getModeName()]['ui']))
 | |
| 			{
 | |
| 				$class_name = $this->modes[$request->getModeName()]['ui'];
 | |
| 				$class_path = vartrue($this->modes[$request->getModeName()]['uipath']);
 | |
| 				if($class_path)
 | |
| 				{
 | |
| 					require_once(e107::getParser()->replaceConstants($class_path));
 | |
| 				}
 | |
| 				if(class_exists($class_name))//NOTE: autoload in the play
 | |
| 				{
 | |
| 					$this->_current_controller->setParam('ui', new $class_name($this->_current_controller));
 | |
| 				}
 | |
| 			}
 | |
| 			$this->_current_controller->setParam('modes', $this->modes);
 | |
| 
 | |
| 		}
 | |
| 		// Not known controller (not found in e_admin_dispatcher::$modes) exception
 | |
| 		else
 | |
| 		{
 | |
| 			// TODO - admin log
 | |
| 			$this->_current_controller = $this->getDefaultController();
 | |
| 			// add messages
 | |
| 			e107::getMessage()->add('Can\'t find class for controller <strong>"'.ucfirst($request->getModeName()).'"</strong>', E_MESSAGE_ERROR)
 | |
| 				->add('Requested: '.e_REQUEST_SELF.'?'.$request->buildQueryString(), E_MESSAGE_DEBUG);
 | |
| 			// go to not found page
 | |
| 			$request->setMode($this->getDefaultControllerName())->setAction('e404');
 | |
| 			$this->_current_controller->setRequest($request)->init();
 | |
| 		}
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Default controller object - needed if controller not found
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function getDefaultController()
 | |
| 	{
 | |
| 		$class_name = $this->getDefaultControllerName();
 | |
| 		return new $class_name($this->getRequest(), $this->getResponse());
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 *  Default controller name - needed if controller not found
 | |
| 	 * @return string name of controller
 | |
| 	 */
 | |
| 	public function getDefaultControllerName()
 | |
| 	{
 | |
| 		return 'e_admin_controller';
 | |
| 	}
 | |
| 
 | |
| 	public function hasPerms($perms)
 | |
| 	{
 | |
| 		return getperms($perms);
 | |
| 	}
 | |
| 
 | |
| 	public function setAccess($access)
 | |
| 	{
 | |
| 		$this->access = $access;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Admin Menu Generator
 | |
| 	 *
 | |
| 	 * @return string|array
 | |
| 	 */
 | |
| 
 | |
| 	public function renderMenu($debug = false)
 | |
| 	{
 | |
| 
 | |
| 		$tp = e107::getParser();
 | |
| 		$adminMenu = $this->getMenuData();
 | |
| 		$var = [];
 | |
| 		$selected = false;
 | |
| 
 | |
| 		// First loop: Build $var without permissions checks
 | |
| 		foreach($adminMenu as $key => $val)
 | |
| 		{
 | |
| 
 | |
| 		//	$parentKey = '';
 | |
| 
 | |
| 			$tmp = explode('/', trim($key, '/'), 2); // mode/action
 | |
| 
 | |
| 
 | |
| 			$isSubItem = !empty($val['group']);
 | |
| 
 | |
| 			if($isSubItem)
 | |
| 			{
 | |
| 				$parentKey = $val['group'] ?? '';
 | |
| 			}
 | |
| 
 | |
| 			$processedItem = $this->processMenuItem($val, $key, $tmp);
 | |
| 			$processedItem['link_id'] = str_replace('/', '-', $key);
 | |
| 
 | |
| 
 | |
| 			if(!empty($val['selected'])) // Custom selector.
 | |
| 			{
 | |
| 				$selected = $val['selected'] === true ? $key : $val['selected'];
 | |
| 			}
 | |
| 			elseif(!empty($processedItem['selected'])) // match detected in 'uri'
 | |
| 			{
 | |
| 				$selected = $processedItem['selected'];
 | |
| 			}
 | |
| 
 | |
| 			if($isSubItem)
 | |
| 			{
 | |
| 				if(empty($var[$parentKey]))
 | |
| 				{
 | |
| 					$var[$parentKey] = [
 | |
| 						'text'      => 'Unknown',
 | |
| 						'image_src' => e_navigation::guessMenuIcon($parentKey),
 | |
| 						'link_id'   => str_replace('/', '-', $parentKey)
 | |
| 					];
 | |
| 				}
 | |
| 
 | |
| 				// Use full key for sub-items to match $adminMenu
 | |
| 				$subKey = $key;
 | |
| 				if(!is_array($var[$parentKey]))
 | |
| 				{
 | |
| 					$var[$parentKey] = [];
 | |
| 				}
 | |
| 
 | |
| 				if(!isset($var[$parentKey]['sub'][$subKey]))
 | |
| 				{
 | |
| 					$var[$parentKey]['sub'][$subKey] = $processedItem;
 | |
| 				}
 | |
| 
 | |
| 				if(!isset($var[$parentKey]['link_class']))
 | |
| 				{
 | |
| 					$var[$parentKey]['link_class'] = '';
 | |
| 				}
 | |
| 
 | |
| 				if(!empty($processedItem['badge']) && strpos($var[$parentKey]['link_class'], 'has-badge') === false)
 | |
| 				{
 | |
| 					$var[$parentKey]['link_class'] .= ' has-badge';
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				if(!isset($var[$key]))
 | |
| 				{
 | |
| 					$var[$key] = $processedItem;
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		if(!$selected)
 | |
| 		{
 | |
| 			$request = $this->getRequest();
 | |
| 			$selected = $request->getMode() . '/' . $request->getAction();
 | |
| 			e107::getMessage()->addDebug('No selected item found, using default: ' . $selected);
 | |
| 		}
 | |
| 
 | |
| 		// Apply permissions restrictions
 | |
| 		$var = $this->restrictMenuAccess($var, $adminMenu);
 | |
| 
 | |
| 
 | |
| 		// Second loop: Handle links and collapse attributes without permissions checks
 | |
| 		foreach($var as $key => &$item)
 | |
| 		{
 | |
| 			if(!empty($item['sub']))
 | |
| 			{
 | |
| 				$item['link'] = '#';
 | |
| 				$item['link_caret'] = true;
 | |
| 				$item['link_data'] = [
 | |
| 					'data-toggle' => 'collapse',
 | |
| 					'data-target' => '#sub-' . $item['link_id'],
 | |
| 					'role'        => 'button'
 | |
| 				];
 | |
| 				$item['sub_class'] = 'collapse';
 | |
| 				$item['caret'] = true;
 | |
| 
 | |
| 				foreach($item['sub'] as $subKey => &$subItem)
 | |
| 				{
 | |
| 					if($selected === $subKey && !empty($subItem['group']))
 | |
| 					{
 | |
| 						$parent = $subItem['group'];
 | |
| 						$var[$parent]['link_data']['aria-expanded'] = 'true';
 | |
| 						$item['sub_class'] = 'collapse in';
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			elseif(!isset($item['link']))
 | |
| 			{
 | |
| 				$tmp = explode('/', trim($key, '/'), 2);
 | |
| 				$item['link'] = e_REQUEST_SELF . '?mode=' . $tmp[0] . '&action=' . ($tmp[1] ?? 'main');
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		if(empty($var))
 | |
| 		{
 | |
| 			return '';
 | |
| 		}
 | |
| 
 | |
| 		$adminMenuAliases = $this->getMenuAliases();
 | |
| 		$selected = vartrue($adminMenuAliases[$selected], $selected);
 | |
| 
 | |
| 		$icon = '';
 | |
| 		$adminMenuIcon = $this->getMenuIcon();
 | |
| 		if(!empty($adminMenuIcon))
 | |
| 		{
 | |
| 			$icon = e107::getParser()->toIcon($adminMenuIcon);
 | |
| 		}
 | |
| 		elseif(deftrue('e_CURRENT_PLUGIN'))
 | |
| 		{
 | |
| 			$icon = e107::getPlug()->load(e_CURRENT_PLUGIN)->getIcon(24);
 | |
| 		}
 | |
| 
 | |
| 		$toggle = "<span class='e-toggle-sidebar'><!-- --></span>";
 | |
| 
 | |
| 		$var['_extras_'] = ['icon' => $icon, 'return' => true];
 | |
| 
 | |
| 		if($debug)
 | |
| 		{
 | |
| 			return $var;
 | |
| 		}
 | |
| 
 | |
| 		return $toggle . e107::getNav()->admin($this->getMenuTitle(), $selected, $var);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Restrict menu items based on permissions and access.
 | |
| 	 *
 | |
| 	 * @param array $var       The menu structure to filter.
 | |
| 	 * @param array $adminMenu The original menu data for reference.
 | |
| 	 * @return array The filtered menu structure.
 | |
| 	 */
 | |
| 	private function restrictMenuAccess($var, $adminMenu)
 | |
| 	{
 | |
| 
 | |
| 		foreach($var as $key => &$item)
 | |
| 		{
 | |
| 			// Check top-level item permissions
 | |
| 			$val = $adminMenu[$key] ?? [];
 | |
| 
 | |
| 			// Handle single-segment keys (e.g., 'treatment') by using the key as the mode
 | |
| 			$mode = strpos($key, '/') !== false ? explode('/', trim($key, '/'), 2)[0] : $key;
 | |
| 
 | |
| 			// Default to true for hasPerms if perm is unset or empty
 | |
| 			$hasPerms = isset($val['perm']) && $val['perm'] !== '' ? $this->hasPerms($val['perm']) : true;
 | |
| 			if(!$hasPerms || !$this->hasModeAccess($mode) || !$this->hasRouteAccess($key))
 | |
| 			{
 | |
| 				unset($var[$key]);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if(!empty($item['sub']))
 | |
| 			{
 | |
| 				$hasValidSubItems = false;
 | |
| 				$parentKey = $key;
 | |
| 				foreach($item['sub'] as $subKey => &$subItem)
 | |
| 				{
 | |
| 					$subVal = $adminMenu[$subKey] ?? [];
 | |
| 					// Log permissions check for sub-item only when removed
 | |
| 					if(!isset($subVal['group']) || $subVal['group'] !== $parentKey)
 | |
| 					{
 | |
| 						unset($item['sub'][$subKey]);
 | |
| 					//	fwrite(STDOUT, "B. restrictMenuAccess: removing subKey=$subKey, parent=$parentKey, group=" . ($subVal['group'] ?? 'none') . ", perm=" . ($subVal['perm'] ?? 'none') . ", hasPerms=" . var_export(isset($subVal['perm']) && $subVal['perm'] !== '' ? $this->hasPerms($subVal['perm']) : true, true) . ", hasModeAccess=" . var_export($this->hasModeAccess($subMode ?? $subKey), true) . ", hasRouteAccess=" . var_export($this->hasRouteAccess($subKey), true) . ", parentRouteAccess=" . var_export($this->hasRouteAccess($parentKey), true) . "\n");
 | |
| 						continue;
 | |
| 					}
 | |
| 					$subMode = strpos($subKey, '/') !== false ? explode('/', trim($subKey, '/'), 2)[0] : $subKey;
 | |
| 					// Default to true for hasPerms if perm is unset or empty
 | |
| 					$hasPerms = isset($subVal['perm']) && $subVal['perm'] !== '' ? $this->hasPerms($subVal['perm']) : true;
 | |
| 					if($hasPerms && $this->hasModeAccess($subMode) && $this->hasRouteAccess($subKey) && $this->hasRouteAccess($parentKey))
 | |
| 					{
 | |
| 						$hasValidSubItems = true;
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						unset($item['sub'][$subKey]);
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				// Remove parent if no valid sub-items or sub-items array is empty
 | |
| 				if(!$hasValidSubItems || empty($item['sub']))
 | |
| 				{
 | |
| 					unset($var[$key]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $var;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $val
 | |
| 	 * @param $key
 | |
| 	 * @param $tmp
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	private function processMenuItem($val, $key, $tmp)
 | |
| 	{
 | |
| 
 | |
| 		$tp = e107::getParser();
 | |
| 		$item = array();
 | |
| 
 | |
| 		foreach($val as $k => $v)
 | |
| 		{
 | |
| 			switch($k)
 | |
| 			{
 | |
| 				case 'caption':
 | |
| 					$k2 = 'text';
 | |
| 					$v = defset($v, $v);
 | |
| 					break;
 | |
| 
 | |
| 				case 'url':
 | |
| 					$k2 = 'link';
 | |
| 					$qry = (isset($val['query'])) ? $val['query'] : '?mode=' . $tmp[0] . '&action=' . ($tmp[1] ?? 'main') . (isset($tmp[2]) ? '&sub=' . $tmp[2] : '');
 | |
| 					$v = $tp->replaceConstants($v, 'abs') . $qry;
 | |
| 					break;
 | |
| 
 | |
| 				case 'uri':
 | |
| 					$k2 = 'link';
 | |
| 					$v = $tp->replaceConstants($v, 'abs');
 | |
| 					if($this->compareUris($v,e_REQUEST_URL))
 | |
| 					{
 | |
| 						$item['selected'] = $key;
 | |
| 					}
 | |
| 					break;
 | |
| 
 | |
| 				case 'badge':
 | |
| 					$k2 = 'badge';
 | |
| 					$v = (array) $v;
 | |
| 					break;
 | |
| 
 | |
| 				case 'icon':
 | |
| 					$k2 = 'image_src';
 | |
| 					$v = (string) $v . '.glyph';  // required, even if empty.
 | |
| 					break;
 | |
| 
 | |
| 				default:
 | |
| 					$k2 = $k;
 | |
| 					break;
 | |
| 			}
 | |
| 
 | |
| 			$item[$k2] = $v;
 | |
| 		}
 | |
| 
 | |
| 		if(!isset($item['image_src']))
 | |
| 		{
 | |
| 			$item['image_src'] = e_navigation::guessMenuIcon($key);
 | |
| 		}
 | |
| 
 | |
| 		if(!empty($item['group'])) // empty icon for group items.
 | |
| 		{
 | |
| 			$item['image_src'] = '.glyph';
 | |
| 		}
 | |
| 
 | |
| 		if(!vartrue($item['link']))
 | |
| 		{
 | |
| 			$item['link'] = e_REQUEST_SELF . '?mode=' . $tmp[0] . '&action=' . ($tmp[1] ?? 'main') . (isset($tmp[2]) ? '&sub=' . $tmp[2] : '');
 | |
| 		}
 | |
| 
 | |
| 		if(varset($val['tab']))
 | |
| 		{
 | |
| 			$item['link'] .= '&tab=' . $val['tab'];
 | |
| 		}
 | |
| 
 | |
| 		if(!empty($val['modal']))
 | |
| 		{
 | |
| 			$item['link_class'] = ' e-modal';
 | |
| 			if(!empty($val['modal-caption']))
 | |
| 			{
 | |
| 				$item['link_data'] = array_merge($item['link_data'] ?? [], ['data-modal-caption' => $val['modal-caption']]);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if(!empty($val['class']))
 | |
| 		{
 | |
| 			$item['link_class'] = ($item['link_class'] ?? '') . ' ' . $val['class'];
 | |
| 		}
 | |
| 
 | |
| 		return $item;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Compares two URIs after normalizing their paths and query parameters.
 | |
| 	 * Specific query parameters ('asc', 'from', 'field') are excluded from comparison.
 | |
| 	 *
 | |
| 	 * @param string $uri1 The first URI to compare.
 | |
| 	 * @param string $uri2 The second URI to compare.
 | |
| 	 * @return bool Returns true if the normalized URIs are equivalent; otherwise, false.
 | |
| 	 */
 | |
| 	private function compareUris($uri1, $uri2)
 | |
| 	{
 | |
| 		if(empty($uri1) || empty($uri2))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		$parsedUri1 = parse_url($uri1);
 | |
| 		$parsedUri2 = parse_url($uri2);
 | |
| 
 | |
| 		$path1 = $parsedUri1['path'] ?? '';
 | |
| 		$path2 = $parsedUri2['path'] ?? '';
 | |
| 
 | |
| 		$query1 = [];
 | |
| 		$query2 = [];
 | |
| 
 | |
| 		if(isset($parsedUri1['query']))
 | |
| 		{
 | |
| 			parse_str($parsedUri1['query'], $query1);
 | |
| 		}
 | |
| 		if(isset($parsedUri2['query']))
 | |
| 		{
 | |
| 			parse_str($parsedUri2['query'], $query2);
 | |
| 		}
 | |
| 
 | |
| 		// Remove asc, from and field from queries
 | |
| 		foreach(['asc', 'from', 'field'] as $param)
 | |
| 		{
 | |
| 			unset($query1[$param], $query2[$param]);
 | |
| 		}
 | |
| 
 | |
| 		$cleanQuery1 = http_build_query($query1);
 | |
| 		$cleanQuery2 = http_build_query($query2);
 | |
| 
 | |
| 		$cleanUri1 = $path1 . ($cleanQuery1 ? '?' . $cleanQuery1 : '');
 | |
| 		$cleanUri2 = $path2 . ($cleanQuery2 ? '?' . $cleanQuery2 : '');
 | |
| 
 | |
| 		return $cleanUri1 === $cleanUri2 ;
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Render Help Text in <ul> format. XXX TODO
 | |
| 	 */
 | |
| 	public function renderHelp()
 | |
| 	{
 | |
| 
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	
 | |
| 	/** 
 | |
| 	 * Check for table issues and warn the user. XXX TODO 
 | |
| 	 * ie. user is using French interface but no french tables found for the current DB tables. 
 | |
| 	 */
 | |
| 	public function renderWarnings()
 | |
| 	{
 | |
| 		
 | |
| 		
 | |
| 		
 | |
| 		
 | |
| 	}
 | |
| 	
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  */
 | |
| class e_admin_controller
 | |
| {
 | |
| 	/**
 | |
| 	 * @var e_admin_request
 | |
| 	 */
 | |
| 	protected $_request;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var e_admin_response
 | |
| 	 */
 | |
| 	protected $_response;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var array User defined parameters
 | |
| 	 */
 | |
| 	protected $_params = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string default action name
 | |
| 	 */
 | |
| 	protected $_default_action = 'index';
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string default trigger action.
 | |
| 	 */
 | |
| 	protected $_default_trigger = 'auto';
 | |
| 	
 | |
| 	/**
 | |
| 	 * List (numerical array) of only allowed for this controller actions
 | |
| 	 * Useful to grant access for certain pre-defined actions only
 | |
| 	 * XXX - we may move this in dispatcher (or even having it also there), still searching the most 'friendly' way
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $allow = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * List (numerical array) of only disallowed for this controller actions
 | |
| 	 * Useful to restrict access for certain pre-defined actions only
 | |
| 	 * XXX - we may move this in dispatcher (or even having it also there), still searching the most 'friendly' way
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $disallow = array();
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor
 | |
| 	 * @param e_admin_request $request
 | |
| 	 */
 | |
| 	public function __construct($request, $response, $params = array())
 | |
| 	{
 | |
| 		$this->_params = array_merge(array('enable_triggers' => false), $params);
 | |
| 		$this->setRequest($request)
 | |
| 			->setResponse($response)
 | |
| 			->setParams($params);
 | |
| 			
 | |
| 		$this->checkAccess();
 | |
| 
 | |
| 		$this->_log(); // clear the log (when debug is enabled)
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Check against allowed/disallowed actions
 | |
| 	 * FIXME check plugin admin access (check_class(P)), confirm e-token is verified
 | |
| 	 */
 | |
| 	public function checkAccess()
 | |
| 	{
 | |
| 		$request = $this->getRequest();
 | |
| 		$currentAction = $request->getAction();
 | |
| 
 | |
| 		// access based on mode setting - general controller access
 | |
| 		if(!empty($this->disallow) && in_array($currentAction, $this->disallow))
 | |
| 		{
 | |
| 			$request->setAction('e403');
 | |
| 			e107::getMessage()->addError(LAN_NO_PERMISSIONS)
 | |
| 				->addDebug('Controller action disallowed restriction triggered.');
 | |
| 			return false;
 | |
| 		}
 | |
| 		
 | |
| 		// access based on $access settings - access per action
 | |
| 		if(!empty($this->allow) && !in_array($currentAction, $this->allow))
 | |
| 		{
 | |
| 			$request->setAction('e403');
 | |
| 			e107::getMessage()->addError(LAN_NO_PERMISSIONS)
 | |
| 				->addDebug('Controller action not in allowed list restriction triggered.');
 | |
| 			return false;
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined init
 | |
| 	 * Called before dispatch routine
 | |
| 	 */
 | |
| 	public function init()
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get controller parameter
 | |
| 	 * Currently used core parameters:
 | |
| 	 * - enable_triggers: don't use it direct, see {@link setTriggersEnabled()}
 | |
| 	 * - modes - see dispatcher::$modes
 | |
| 	 * - ajax_response - text|xml|json - default is 'text'; this should be set by the action method
 | |
| 	 * - TODO - more parameters/add missing to this list
 | |
| 	 *
 | |
| 	 * @param string $key [optional] if null - get whole array
 | |
| 	 * @param mixed $default [optional]
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function getParam($key = null, $default = null)
 | |
| 	{
 | |
| 		if($key === null)
 | |
| 		{
 | |
| 			return $this->_params;
 | |
| 		}
 | |
| 		return (isset($this->_params[$key]) ? $this->_params[$key] : $default);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set parameter
 | |
| 	 * @param string $key
 | |
| 	 * @param mixed $value
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setParam($key, $value)
 | |
| 	{
 | |
| 		if($value === null)
 | |
| 		{
 | |
| 			unset($this->_params[$key]);
 | |
| 			return $this;
 | |
| 		}
 | |
| 		$this->_params[$key] = $value;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Merge passed parameter array with current parameters
 | |
| 	 * @param array $params
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setParams($params)
 | |
| 	{
 | |
| 		$this->_params = array_merge($this->_params, $params);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reset parameter array
 | |
| 	 * @param array $params
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function resetParams($params)
 | |
| 	{
 | |
| 		$this->_params = $params;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current request object
 | |
| 	 * @return e_admin_request
 | |
| 	 */
 | |
| 	public function getRequest()
 | |
| 	{
 | |
| 		return $this->_request;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set current request object
 | |
| 	 * @param e_admin_request $request
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setRequest($request)
 | |
| 	{
 | |
| 		$this->_request = $request;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current response object
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function getResponse()
 | |
| 	{
 | |
| 		return $this->_response;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set current response object
 | |
| 	 * @param e_admin_response $response
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setResponse($response)
 | |
| 	{
 | |
| 		$this->_response = $response;
 | |
| 		return $this;
 | |
| 	}
 | |
| 	
 | |
| 	/**
 | |
| 	 * Get current dispatcher object
 | |
| 	 * @return e_admin_dispatcher
 | |
| 	 */
 | |
| 	public function getDispatcher()
 | |
| 	{
 | |
| 		return e107::getRegistry('admin/ui/dispatcher');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Request proxy method
 | |
| 	 * @param string $key [optional]
 | |
| 	 * @param mixed $default [optional]
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function getQuery($key = null, $default = null)
 | |
| 	{
 | |
| 		return $this->getRequest()->getQuery($key, $default);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Request proxy method
 | |
| 	 * @param string|array $key
 | |
| 	 * @param mixed $value [optional]
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setQuery($key, $value = null)
 | |
| 	{
 | |
| 		$this->getRequest()->setQuery($key, $value);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Request proxy method
 | |
| 	 * @param string $key [optional]
 | |
| 	 * @param mixed $default [optional]
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function getPosted($key = null, $default = null)
 | |
| 	{
 | |
| 		return $this->getRequest()->getPosted($key, $default);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Request proxy method
 | |
| 	 * @param string|array $key
 | |
| 	 * @param mixed $value [optional]
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setPosted($key, $value = null)
 | |
| 	{
 | |
| 		$this->getRequest()->setPosted($key, $value);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add page title, response proxy method
 | |
| 	 *
 | |
| 	 * @param string $title if boolean true - current menu caption will be used
 | |
| 	 * @param boolean $meta add to meta as well
 | |
| 	 * @return object e_admin_controller
 | |
| 	 */
 | |
| 	public function addTitle($title = true, $meta = true)
 | |
| 	{
 | |
| 		
 | |
| 		$response = $this->getResponse();
 | |
| 
 | |
| 		if($title === true)
 | |
| 		{
 | |
| 			$_dispatcher = $this->getDispatcher();
 | |
| 			$data = $_dispatcher->getPageTitles();
 | |
| 			$search = $this->getMode().'/'.$this->getAction();
 | |
| 
 | |
| 
 | |
| 
 | |
| 			if(isset($data[$search]))
 | |
| 			{
 | |
| 				 $res['caption'] = $data[$search];
 | |
| 			}
 | |
| 			else 
 | |
| 			{
 | |
| 
 | |
| 
 | |
| 				$data = $_dispatcher->getMenuData();
 | |
| 
 | |
| 				if(isset($data[$search]))
 | |
| 				{
 | |
| 					 $res = $data[$search];
 | |
| 					 if(!empty($res['group']))
 | |
| 					 {
 | |
| 					    $parent = $res['group'];
 | |
| 						$parentCaption = $data[$parent]['caption'] ?? '';
 | |
| 						if(!empty($parentCaption))
 | |
| 						{
 | |
| 							$response->appendTitle($parentCaption);
 | |
| 						}
 | |
| 
 | |
| 					 }
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					// check for an alias match.
 | |
| 					$d = $_dispatcher->getMenuAliases();
 | |
| 					if(isset($d[$search]))
 | |
| 					{
 | |
| 						$search = $d[$search];
 | |
| 						$res = $data[$search];
 | |
| 
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						 return $this;
 | |
| 					}
 | |
| 				//	var_dump($d);
 | |
| 				//	var_dump("Couldnt find: ".$search);
 | |
| 
 | |
| 				}
 | |
| 			}
 | |
| 			$title = defset($res['caption'],$res['caption']);
 | |
| 
 | |
| 
 | |
| 		}
 | |
| 		
 | |
| 		//	echo "<h3>".__METHOD__." - ".$title."</h3>";
 | |
| 	
 | |
| 	//	print_a($title);
 | |
| 		$response->appendTitle($title);
 | |
| 
 | |
| 		if($meta)
 | |
| 		{
 | |
| 			$this->addMetaTitle($title);
 | |
| 		}
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add page meta title, response proxy method.
 | |
| 	 * Should be called before header.php
 | |
| 	 *
 | |
| 	 * @param string $title
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function addMetaTitle($title=null)
 | |
| 	{
 | |
| 		if($title === null)
 | |
| 		{
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		$this->getResponse()->addMetaTitle($title);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Add header content, response proxy method
 | |
| 	 * Should be called before header.php
 | |
| 	 *
 | |
| 	 * @param string $content
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function addHeader($content=null)
 | |
| 	{
 | |
| 		if($content === null)
 | |
| 		{
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		$this->getResponse()->addHeaderContent(vartrue($content));
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get header content, response proxy method
 | |
| 	 *
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getHeader()
 | |
| 	{
 | |
| 		return $this->getResponse()->getHeaderContent();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current mode, response proxy method
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getMode()
 | |
| 	{
 | |
| 		return $this->getRequest()->getMode();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current actin, response proxy method
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getAction()
 | |
| 	{
 | |
| 		return $this->getRequest()->getAction();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current ID, response proxy method
 | |
| 	 * @return int
 | |
| 	 */
 | |
| 	public function getId()
 | |
| 	{
 | |
| 		return $this->getRequest()->getId();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get response owned JS Helper instance, response proxy method
 | |
| 	 *
 | |
| 	 * @return e_jshelper
 | |
| 	 */
 | |
| 	public function getJsHelper()
 | |
| 	{
 | |
| 		return $this->getResponse()->getJsHelper();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $action
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function _preDispatch($action = '')
 | |
| 	{
 | |
| 		if(!$action)
 | |
| 		{
 | |
| 			$action = $this->getRequest()->getActionName();
 | |
| 		}
 | |
| 		$method = $this->toMethodName($action);
 | |
| 		if(!method_exists($this, $method))
 | |
| 		{
 | |
| 			$this->_log('Skipping ' .$method. '() (not found)');
 | |
| 			$this->getRequest()->setAction($this->getDefaultAction());
 | |
| 		}
 | |
| 
 | |
| 		// switch to 404 if needed
 | |
| 		$method = $this->toMethodName($this->getRequest()->getActionName());
 | |
| 		if(!method_exists($this, $method))
 | |
| 		{
 | |
| 			$this->_log('Skipping ' .$method. '() (not found)');
 | |
| 			$this->getRequest()->setAction('e404');
 | |
| 			$message = e107::getParser()->lanVars(LAN_UI_404_METHOD_ERROR, $method, true);
 | |
| 			e107::getMessage()->add($message, E_MESSAGE_ERROR);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Log Controller when e_DEBUG is active.
 | |
| 	 * @param string|null $message
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	protected function _log($message=null)
 | |
| 	{
 | |
| 		if(!deftrue('e_DEBUG'))
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		if($message === null) // clear the log.
 | |
| 		{
 | |
| 			file_put_contents(e_LOG. 'adminUI.log', '');
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		$date = (!empty($message)) ? date('c') : '';
 | |
| 
 | |
| 		file_put_contents(e_LOG. 'adminUI.log',$date."\t".$message."\n",FILE_APPEND);
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Dispatch observer, check for triggers
 | |
| 	 *
 | |
| 	 * @param string $action [optional]
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function dispatchObserver($action = null)
 | |
| 	{
 | |
| 		$request = $this->getRequest();
 | |
| 		if($request === null)
 | |
| 		{
 | |
| 			$request = new e_admin_request();
 | |
| 			$this->setRequest($request);
 | |
| 		}
 | |
| 
 | |
| 		$this->_preDispatch($action);
 | |
| 		if($action === null)
 | |
| 		{
 | |
| 			$action = $request->getActionName();
 | |
| 		}
 | |
| 
 | |
| 		// check for observer
 | |
| 		$actionObserverName = $this->toMethodName($action, 'observer', e_AJAX_REQUEST);
 | |
| 		if(method_exists($this, $actionObserverName))
 | |
| 		{
 | |
| 			$this->_log('Executing ' .$actionObserverName. '()');
 | |
| 			$this->$actionObserverName();
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$this->_log('Skipping ' .$actionObserverName. '() (not found)');
 | |
| 		}
 | |
| 
 | |
| 		// check for triggers, not available in Ajax mode
 | |
| 		$triggerEnabled = $this->triggersEnabled();
 | |
| 		if(!e_AJAX_REQUEST && $triggerEnabled)
 | |
| 		{
 | |
| 			if($posted = $request->getPosted())
 | |
| 			{
 | |
| 				foreach ($posted as $key => $value)
 | |
| 				{
 | |
| 					if(strpos($key, 'etrigger_') === 0)
 | |
| 					{
 | |
| 						$actionTriggerName = $this->toMethodName($action.$request->camelize(substr($key, 9)), 'trigger', false);
 | |
| 						if(method_exists($this, $actionTriggerName))
 | |
| 						{
 | |
| 							$this->$actionTriggerName($value);
 | |
| 						}
 | |
| 						//Check if triggers are still enabled
 | |
| 						if(!$triggerEnabled)
 | |
| 						{
 | |
| 							break;
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Dispatch header, not allowed in Ajax mode
 | |
| 	 * @param string $action [optional]
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function dispatchHeader($action = null)
 | |
| 	{
 | |
| 		// not available in Ajax mode
 | |
| 		if(e_AJAX_REQUEST)
 | |
| 		{
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		$request = $this->getRequest();
 | |
| 		if($request === null)
 | |
| 		{
 | |
| 			$request = new e_admin_request();
 | |
| 			$this->setRequest($request);
 | |
| 		}
 | |
| 
 | |
| 		$this->_preDispatch($action);
 | |
| 		if($action === null)
 | |
| 		{
 | |
| 			$action = $request->getActionName();
 | |
| 		}
 | |
| 
 | |
| 		// check for observer
 | |
| 		$actionHeaderName = $this->toMethodName($action, 'header', false);
 | |
| 		if(method_exists($this, $actionHeaderName))
 | |
| 		{
 | |
| 			$this->$actionHeaderName();
 | |
| 		}
 | |
| 
 | |
| 		//send meta data
 | |
| 		$this->getResponse()->sendMeta();
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Dispatch controller action
 | |
| 	 *
 | |
| 	 * @param string $action [optional]
 | |
| 	 * @return e_admin_response
 | |
| 	 */
 | |
| 	public function dispatchPage($action = null)
 | |
| 	{
 | |
| 		$request = $this->getRequest();
 | |
| 		if($request === null)
 | |
| 		{
 | |
| 			$request = new e_admin_request();
 | |
| 			$this->setRequest($request);
 | |
| 		}
 | |
| 		$response = $this->getResponse();
 | |
| 	//	print_a($response);
 | |
| 		$this->_preDispatch($action);
 | |
| 
 | |
| 		if($action === null)
 | |
| 		{
 | |
| 			$action = $request->getActionName();
 | |
| 		}
 | |
| 
 | |
| 		// check for observer
 | |
| 		$actionName = $this->toMethodName($action);
 | |
| 		$ret = '';
 | |
| 		if(!method_exists($this, $actionName)) // pre dispatch already switched to default action/not found page if needed
 | |
| 		{
 | |
| 			$this->_log('Skipping ' .$actionName. '() (not found)');
 | |
| 			e107::getMessage()->add('Action '.$actionName.' no found!', E_MESSAGE_ERROR);
 | |
| 			return $response;
 | |
| 		}
 | |
| 
 | |
| 		$this->_log('Executing ' .$actionName. '()');
 | |
| 
 | |
| 		if($action !== 'Prefs' && $action !== 'Create' && $action !== 'Edit' && $action !== 'List') // Custom Page method in use, so add the title.
 | |
| 		{
 | |
| 			$this->addTitle();
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 	//	e107::getDebug()->log("Admin-ui Action: <b>".$action."</b>");
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 		
 | |
| 		
 | |
| 		ob_start(); //catch any output
 | |
| 		$ret = $this->{$actionName}();
 | |
| 
 | |
| 
 | |
| 		//Ajax XML/JSON communication
 | |
| 		if(e_AJAX_REQUEST && is_array($ret))
 | |
| 		{
 | |
| 			$response_type = $this->getParam('ajax_response', 'xml');
 | |
| 			ob_clean();
 | |
| 			$js_helper = $response->getJsHelper();
 | |
| 			foreach ($ret as $act => $data)
 | |
| 			{
 | |
| 				$js_helper->addResponse($data, $act);
 | |
| 			}
 | |
| 			$js_helper->sendResponse($response_type);
 | |
| 		}
 | |
| 
 | |
| 		$ret .= ob_get_clean();
 | |
| 
 | |
| 		// Ajax text response
 | |
| 		if(e_AJAX_REQUEST)
 | |
| 		{
 | |
| 			$response_type = 'text';
 | |
| 			$response->getJsHelper()->addResponse($ret)->sendResponse($response_type);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$response->appendBody($ret);
 | |
| 		}
 | |
| 	
 | |
| 		return $response;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function E404Observer()
 | |
| 	{
 | |
| 		$this->getResponse()->setTitle(LAN_UI_404_TITLE_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function E404Page()
 | |
| 	{
 | |
| 		return '<div class="center">'.LAN_UI_404_BODY_ERROR.'</div>';
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function E404AjaxPage()
 | |
| 	{
 | |
| 		exit;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function E403Observer()
 | |
| 	{
 | |
| 		$this->getResponse()->setTitle(LAN_UI_403_TITLE_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function E403Page()
 | |
| 	{
 | |
| 		return '<div class="center">'.LAN_UI_403_BODY_ERROR.'</div>';
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function E403AjaxPage()
 | |
| 	{
 | |
| 		exit;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic redirect handler, it handles almost everything we would need.
 | |
| 	 * Additionally, it moves currently registered system messages to SESSION message stack
 | |
| 	 * In almost every case {@link redirectAction()} and {@link redirectMode()} are better solution
 | |
| 	 *
 | |
| 	 * @param string $action defaults to current action
 | |
| 	 * @param string $mode defaults to current mode
 | |
| 	 * @param string|array $exclude_query comma delimited variable names to be excluded from current query OR TRUE to exclude everything
 | |
| 	 * @param string|array $merge_query query string (& delimiter) or associative array to be merged with current query
 | |
| 	 * @param string $path default to e_SELF
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function redirect($action = null, $mode = null, $exclude_query = '', $merge_query = array(), $path = null)
 | |
| 	{
 | |
| 		$request = $this->getRequest();
 | |
| 
 | |
| 		if($mode)
 | |
| 		{
 | |
| 			$request->setMode($mode);
 | |
| 		}
 | |
| 		if($action)
 | |
| 		{
 | |
| 			$request->setAction($action);
 | |
| 		}
 | |
| 		if(!$path)
 | |
| 		{
 | |
| 			$path = e_REQUEST_SELF;
 | |
| 		}
 | |
| 		
 | |
| 		//prevent cache
 | |
| 		header('Cache-Control: private, no-store, no-cache, must-revalidate, post-check=0, pre-check=0');
 | |
| 	//	header('Pragma: no-cache');
 | |
| 
 | |
| 		$url = $path.'?'.$request->buildQueryString($merge_query, false, $exclude_query);
 | |
| 		// Transfer all messages to session
 | |
| 		e107::getMessage()->moveToSession();
 | |
| 		// write session data
 | |
| 		session_write_close();
 | |
| 
 | |
| 		// do redirect
 | |
| 		e107::redirect($url);
 | |
| 	//	header('Location: '.$url);
 | |
| 		exit;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Convenient redirect() proxy method, make life easier when redirecting between actions
 | |
| 	 * in same mode.
 | |
| 	 *
 | |
| 	 * @param string $action [optional]
 | |
| 	 * @param string|array $exclude_query [optional]
 | |
| 	 * @param string|array $merge_query [optional]
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function redirectAction($action = null, $exclude_query = '', $merge_query = array())
 | |
| 	{
 | |
| 		$this->redirect($action, null, $exclude_query, $merge_query);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Convenient redirect to another mode (doesn't use current Query state)
 | |
| 	 * If path is empty, it'll be auto-detected from modes (dispatcher) array
 | |
| 	 *
 | |
| 	 * @param string $mode
 | |
| 	 * @param string $action
 | |
| 	 * @param string|array $query [optional]
 | |
| 	 * @param string $path
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function redirectMode($mode, $action, $query = array(), $path = null)
 | |
| 	{
 | |
| 		if(!$path && $this->getParam('modes'))
 | |
| 		{
 | |
| 			$modes = $this->getParam('modes');
 | |
| 			if(!empty($modes[$mode]) && !empty($modes[$mode]['url']))
 | |
| 			{
 | |
| 				$path = e107::getParser()->replaceConstants($modes[$mode]['url'], 'abs');
 | |
| 			}
 | |
| 		}
 | |
| 		$this->redirect($action, $mode, true, $query, $path);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Convert action name to method name
 | |
| 	 *
 | |
| 	 * @param string $action_name formatted (e.g. request method getActionName()) action name
 | |
| 	 * @param string $type page|observer|header|trigger
 | |
| 	 * @param boolean $ajax force with true/false, if null will be auto-resolved
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function toMethodName($action_name, $type= 'page', $ajax = null)
 | |
| 	{
 | |
| 		if($ajax === null)
 | |
| 		{
 | |
| 			$ajax = e_AJAX_REQUEST;
 | |
| 		} //auto-resolving
 | |
| 		return $action_name.($ajax ? 'Ajax' : '').ucfirst(strtolower($type));
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Check if there is a trigger available in the posted data
 | |
| 	 * @param array $exclude
 | |
| 	 * @return boolean
 | |
| 	 */
 | |
| 	public function hasTrigger($exclude = array())
 | |
| 	{
 | |
| 		$posted = array_keys($this->getPosted());
 | |
| 		foreach ($posted as $key)
 | |
| 		{
 | |
| 			if(!in_array($key, $exclude) && strpos($key, 'etrigger_') === 0)
 | |
| 			{
 | |
| 				return true;
 | |
| 			}
 | |
| 		}
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get default action
 | |
| 	 * @return string action
 | |
| 	 */
 | |
| 	public function getDefaultAction()
 | |
| 	{
 | |
| 		return $this->_default_action;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getDefaultTrigger()
 | |
| 	{
 | |
| 		return $this->_default_trigger;
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set default action
 | |
| 	 * @param string $action_name
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setDefaultAction($action_name)
 | |
| 	{
 | |
| 		$this->_default_action = $action_name;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 	 * Set default trigger
 | |
| 	 * @param string|array $triggers 'auto' or array of triggers
 | |
| 	 * @example $triggers['submit'] = array(LAN_UPDATE, 'update', $model->getId());
 | |
| 				$triggers['submit'] = array(LAN_CREATE, 'create', 0);
 | |
| 				$triggers['cancel'] = array(LAN_CANCEL, 'cancel');
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setDefaultTrigger($triggers)
 | |
| 	{
 | |
| 		$this->_default_trigger = $triggers;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return boolean
 | |
| 	 */
 | |
| 	public function triggersEnabled()
 | |
| 	{
 | |
| 		return $this->getParam('enable_triggers');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param boolean $flag
 | |
| 	 * @return e_admin_controller
 | |
| 	 */
 | |
| 	public function setTriggersEnabled($flag)
 | |
| 	{
 | |
| 		$this->setParam('enable_triggers', $flag);
 | |
| 		return $this;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| //FIXME - move everything from e_admin_ui except model auto-create related code
 | |
| 
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  */
 | |
| class e_admin_controller_ui extends e_admin_controller
 | |
| {
 | |
| 
 | |
| 	protected $table;
 | |
| 	/**
 | |
| 	 * @var array UI field data
 | |
| 	 */
 | |
| 
 | |
| 	/** @var string  */
 | |
| 	protected $listQry;
 | |
| 
 | |
| 	protected $pid;
 | |
| 
 | |
| 	protected $fields = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * @var array default fields activated on List view
 | |
| 	 */
 | |
| 	protected $fieldpref = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Custom Field (User) Preferences Name. (for viewable columns)
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $fieldPrefName = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * @var array Plugin Preference description array
 | |
| 	 */
 | |
| 	protected $prefs = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Data required for _modifyListQry() to automate
 | |
| 	 * db query building
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $tableJoin = array(); 
 | |
| 
 | |
| 	/**
 | |
| 	 * Array of table names and their aliases. (detected from listQry)
 | |
| 	 * db query building
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $joinAlias = array(); 
 | |
| 
 | |
| 	/**
 | |
| 	 * Array of fields detected from listQry which are JOINs
 | |
| 	 * @example returns array('user_name'=>'u.user_name'); from $listQry = "SELECT n.*,u.user_name FROM #news...."etc.
 | |
| 	 */
 | |
| 	protected $joinField = array();
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Main model table alias
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $tableAlias;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string plugin name
 | |
| 	 */
 | |
| 	protected $pluginName;
 | |
| 	
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string event name
 | |
| 	 * base event trigger name to be used. Leave blank for no trigger. 
 | |
| 	 */
 | |
| 	protected $eventName;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $defaultOrderField;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	protected $defaultOrder = 'asc';
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string SQL order, false to disable order, null is default order
 | |
| 	 */
 | |
| 	protected $listOrder;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string SQL group-by field name (optional)
 | |
| 	 */
 | |
| 	protected $listGroup;
 | |
| 	
 | |
| 	/**
 | |
| 	 * @var string field containing the order number
 | |
| 	 */
 | |
| 	protected $sortField;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string field containing the order number
 | |
| 	 */
 | |
| 	protected $treePrefix;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var string field containing the parent field
 | |
| 	 */
 | |
| 	protected $sortParent;
 | |
| 	
 | |
| 	/**
 | |
| 	 * @var int reorder step
 | |
| 	 */
 | |
| 	protected $orderStep = 1;
 | |
| 	
 | |
| 	/**
 | |
| 	 * Example: array('0' => 'Tab label', '1' => 'Another label');
 | |
| 	 * Referenced from $field property per field - 'tab => xxx' where xxx is the tab key (identifier)
 | |
| 	 * @var array edit/create form tabs
 | |
| 	 */
 | |
| 	protected $tabs = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * Example: array('0' => 'Tab label', '1' => 'Another label');
 | |
| 	 * Referenced from $prefs property per field - 'tab => xxx' where xxx is the tab key (identifier)
 | |
| 	 * @var array edit/create form tabs
 | |
| 	 */
 | |
| 	protected $preftabs = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * TODO Example: 
 | |
| 	 * Contains required data for auto-assembling URL from every record
 | |
| 	 * For greater control - override url() method
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $url = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * TODO Example: 
 | |
| 	 * Contains required data for mapping featurebox fields
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $featurebox = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Structure same as TreeModel parameters used for building the load() SQL
 | |
| 	 * @var additional SQL to be applied when auto-building the list query
 | |
| 	 */
 | |
| 	protected $listQrySql = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * @var Custom Filter SQL Query override.
 | |
| 	 */
 | |
| 	protected $filterQry;
 | |
| 
 | |
| 	/**
 | |
| 	 * Custom sorting of the filter list.
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $filterSort = array();
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Custom sorting of the batch list.
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $batchSort = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * @var boolean
 | |
| 	 */
 | |
| 	protected $batchDelete = true;
 | |
| 	
 | |
| 	/**
 | |
| 	 * @var boolean
 | |
| 	 */
 | |
| 	protected $batchCopy = false;
 | |
| 	
 | |
|     /**
 | |
|      * @var boolean
 | |
|      */
 | |
|     protected $batchLink = false;
 | |
| 	
 | |
|     /**
 | |
|      * @var boolean
 | |
|      */
 | |
|     protected $batchFeaturebox = false;
 | |
| 
 | |
| 	 /**
 | |
|      * @var boolean
 | |
|      */
 | |
|     protected $batchExport = false;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $batchOptions = array();
 | |
| 	
 | |
| 	/**
 | |
| 	 * Could be LAN constant (mulit-language support)
 | |
| 	 *
 | |
| 	 * @var string plugin name
 | |
| 	 */
 | |
| 	protected $pluginTitle;
 | |
| 
 | |
| 	/**
 | |
| 	 * Default (db) limit value
 | |
| 	 * @var integer
 | |
| 	 */
 | |
| 	protected $perPage = 20;
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Data for grid layout.
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $grid = array();
 | |
| 	
 | |
| 		/**
 | |
| 	 * @var e_admin_model
 | |
| 	 */
 | |
| 	protected $formQuery = false; // custom form post query
 | |
| 
 | |
| 	/**
 | |
| 	 * @var e_admin_model
 | |
| 	 */
 | |
| 	protected $_model;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var e_admin_tree_model
 | |
| 	 */
 | |
| 	protected $_tree_model;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var e_admin_form_ui
 | |
| 	 */
 | |
| 	protected $_ui;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var e_plugin_pref|e_core_pref
 | |
| 	 */
 | |
| 	protected $_pref;
 | |
| 	
 | |
| 	/**
 | |
| 	 * Prevent parsing table aliases more than once
 | |
| 	 * @var boolean
 | |
| 	 */
 | |
| 	protected $_alias_parsed = false;
 | |
| 
 | |
| 	/**
 | |
| 	 * @var bool
 | |
| 	 */
 | |
| 	protected $afterSubmitOptions = true;
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function getAfterSubmitOptions()
 | |
| 	{
 | |
| 		return $this->afterSubmitOptions;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function getBatchDelete()
 | |
| 	{
 | |
| 		return $this->batchDelete;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function getBatchCopy()
 | |
| 	{
 | |
| 		return $this->batchCopy;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function getBatchLink()
 | |
|     {
 | |
|         return $this->batchLink;
 | |
|     }
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function getBatchFeaturebox()
 | |
|     {
 | |
|         return $this->batchFeaturebox;
 | |
|     }
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function getBatchExport()
 | |
|     {
 | |
|         return $this->batchExport;
 | |
|     }
 | |
| 
 | |
| 	/**
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getBatchOptions()
 | |
| 	{
 | |
| 		return $this->batchOptions;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getEventName()
 | |
| 	{
 | |
| 		return  $this->eventName;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getFilterSort()
 | |
| 	{
 | |
| 		return  $this->filterSort;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getBatchSort()
 | |
| 	{
 | |
| 		return  $this->batchSort;
 | |
| 	}
 | |
| 	
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getPluginName()
 | |
| 	{
 | |
| 		return $this->pluginName;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getPluginTitle()
 | |
| 	{
 | |
| 		return deftrue($this->pluginTitle, $this->pluginTitle);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get Sort Field data
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getSortField()
 | |
| 	{
 | |
| 		return $this->sortField;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get Sort Field data
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getSortParent()
 | |
| 	{
 | |
| 		return $this->sortParent;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 	 * Get Sort Field data
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getTreePrefix()
 | |
| 	{
 | |
| 		return $this->treePrefix;
 | |
| 	}
 | |
| 	
 | |
| 	/**
 | |
| 	 * Get Tab data
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getTabs()
 | |
| 	{
 | |
| 		return $this->tabs;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $key
 | |
| 	 * @param $val
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function addTab($key, $val)
 | |
| 	{
 | |
| 		$this->tabs[$key] = (string) $val;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get Tab data
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getPrefTabs()
 | |
| 	{
 | |
| 		return $this->preftabs;
 | |
| 	}
 | |
| 
 | |
|     /**
 | |
|      * Get URL profile
 | |
|      * @return array
 | |
|      */
 | |
|     public function getUrl()
 | |
|     {
 | |
|         return $this->url;
 | |
|     }
 | |
| 	
 | |
| 
 | |
|       /**
 | |
|      * Get Featurebox Copy 
 | |
|      * @return array
 | |
|      */
 | |
|     public function getFeaturebox()
 | |
|     {
 | |
|         return $this->featurebox;
 | |
|     }
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Get all field data
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getFields()
 | |
| 	{
 | |
| 		return $this->fields;
 | |
| 	}
 | |
| 
 | |
| 	public function setFields($fields)
 | |
| 	{
 | |
| 		$this->fields = $fields;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 *
 | |
| 	 * @param string $field
 | |
| 	 * @param string $key attribute name
 | |
| 	 * @param mixed $default default value if not set, default is null
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function getFieldAttr($field, $key = null, $default = null)
 | |
| 	{
 | |
| 		if(isset($this->fields[$field]))
 | |
| 		{
 | |
| 			if($key !== null)
 | |
| 			{
 | |
| 				return isset($this->fields[$field][$key]) ? $this->fields[$field][$key] : $default;
 | |
| 			}
 | |
| 			return $this->fields[$field];
 | |
| 		}
 | |
| 		return $default;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 *
 | |
| 	 * @param string|array $field
 | |
| 	 * @param string $key attribute name
 | |
| 	 * @param mixed $value default value if not set, default is null
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	public function setFieldAttr($field, $key = null, $value = null)
 | |
| 	{
 | |
| 		// add field array
 | |
| 		if(is_array($field))
 | |
| 		{
 | |
| 			foreach ($field as $f => $atts)
 | |
| 			{
 | |
| 				$this->setFieldAttr($f, $atts);
 | |
| 			}
 | |
| 			return $this;
 | |
| 		}
 | |
| 		// remove a field
 | |
| 		if($key === null)
 | |
| 		{
 | |
| 			unset($this->fields[$field]);
 | |
| 			return $this;
 | |
| 		}
 | |
| 		// add to attribute array of a field
 | |
| 		if(is_array($key))
 | |
| 		{
 | |
| 			foreach ($key as $k => $att)
 | |
| 			{
 | |
| 				$this->setFieldAttr($field, $k, $att);
 | |
| 			}
 | |
| 			return $this;
 | |
| 		}
 | |
| 		// remove attribute from field attribute set
 | |
| 		if($value === null && $key !== 'type')
 | |
| 		{
 | |
| 			unset($this->fields[$field][$key]);
 | |
| 			return $this;
 | |
| 		}
 | |
| 		// set attribute value
 | |
| 		$this->fields[$field][$key] = $value;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get fields stored as user preferences
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getFieldPref()
 | |
| 	{
 | |
| 		return $this->fieldpref;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get Config data array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getPrefs()
 | |
| 	{
 | |
| 		return $this->prefs;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return array|int|mixed
 | |
| 	 */
 | |
| 	public function getPerPage()
 | |
| 	{
 | |
| 
 | |
| 		if($this->getAction() === 'grid')
 | |
| 		{
 | |
| 			if($this->getGrid('carousel') === true)
 | |
| 			{
 | |
| 				return 0;
 | |
| 			}
 | |
| 
 | |
| 			return $this->getGrid('perPage');
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		return $this->perPage;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $key
 | |
| 	 * @return array|mixed
 | |
| 	 */
 | |
| 	public function getGrid($key=null)
 | |
| 	{
 | |
| 		if($key !== null)
 | |
| 		{
 | |
| 			return $this->grid[$key];
 | |
| 		}
 | |
| 
 | |
| 		return $this->grid;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return bool|e_admin_model
 | |
| 	 */
 | |
| 	public function getFormQuery()
 | |
| 	{
 | |
| 		return $this->formQuery;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getPrimaryName()
 | |
| 	{
 | |
| 		return $this->getModel()->getFieldIdName();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getDefaultOrderField()
 | |
| 	{
 | |
| 		return ($this->defaultOrder ? $this->defaultOrderField : $this->getPrimaryName());
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getDefaultOrder()
 | |
| 	{
 | |
| 		return ($this->defaultOrder ? $this->defaultOrder : 'asc');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get column preference array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getUserPref()
 | |
| 	{
 | |
| 		//global $user_pref;
 | |
| 		// return vartrue($user_pref['admin_cols_'.$this->getTableName()], array());
 | |
| 
 | |
| 		$name = (!empty($this->fieldPrefName)) ? strtolower($this->pluginName. '_' .$this->fieldPrefName) : $this->getTableName();
 | |
| 
 | |
| 		e107::getMessage()->addDebug('Loading Field Preferences using name: ' .$name);
 | |
| 		$this->_log('Loading Field Preferences using name: ' .$name);
 | |
| 		return e107::getUser()->getPref('admin_cols_'.$name, array());
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set column preference array
 | |
| 	 * @return boolean success
 | |
| 	 */
 | |
| 	public function setUserPref($new, $name='')
 | |
| 	{
 | |
| 		//global $user_pref;
 | |
| 		//e107::getUser()->getConfig()->setData($new);
 | |
| 		//$user_pref['admin_cols_'.$this->getTableName()] = $new;
 | |
| 		//$this->fieldpref = $new;
 | |
| 		//return save_prefs('user');
 | |
| 		if(!empty($new))
 | |
|         {
 | |
| 		    $this->fieldpref = $new;
 | |
|         }
 | |
| 
 | |
| 		if(empty($name))
 | |
| 		{
 | |
| 			$name = $this->getTableName();
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$name = strtolower($this->pluginName. '_' .$name);
 | |
| 		}
 | |
| 
 | |
|         $msg = 'Saving User Field preferences using name: ' .$name;
 | |
| 		e107::getMessage()->addDebug($msg);
 | |
| 		$this->_log($msg);
 | |
| 
 | |
| 		return e107::getUser()->getConfig()
 | |
| 			->set('admin_cols_'.$name, $new)
 | |
| 			->save();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current model
 | |
| 	 *
 | |
| 	 * @return e_admin_model
 | |
| 	 */
 | |
| 	public function getModel()
 | |
| 	{
 | |
| 		if($this->_model === null)
 | |
| 		{
 | |
| 			$this->_setModel();
 | |
| 		}
 | |
| 
 | |
| 		return $this->_model;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Alias for getModel()->get and getListModel()->get().
 | |
| 	 * May be used inside field-method in read/write mode.
 | |
| 	 *
 | |
| 	 * @param string $key
 | |
| 	 * @return mixed|null - current value of the chosen db field.
 | |
| 	 */
 | |
| 	public function getFieldVar($key = null)
 | |
| 	{
 | |
| 		if(empty($key))
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		$action = $this->getAction();
 | |
| 		if($action === 'list' || $action === 'grid')
 | |
| 		{
 | |
| 			$obj = $this->getListModel();
 | |
| 			if(is_object($obj))
 | |
| 			{
 | |
| 				return $obj->get($key);
 | |
| 			}
 | |
| 
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		return $this->getModel()->get($key);
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Set controller model
 | |
| 	 * @param e_admin_model $model
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	public function setModel($model)
 | |
| 	{
 | |
| 		$this->_model = $model;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get model validation array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getValidationRules()
 | |
| 	{
 | |
| 		return $this->getModel()->getValidationRules();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get model data field array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getDataFields()
 | |
| 	{
 | |
| 		return $this->getModel()->getDataFields();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get model table or alias
 | |
| 	 * @param boolean $alias get table alias on true, default false
 | |
| 	 * @param object $prefix add e107 special '#' prefix, default false
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getTableName($alias = false, $prefix = false)
 | |
| 	{
 | |
| 		if($alias)
 | |
| 		{
 | |
| 			return ($this->tableAlias ? $this->tableAlias : '');
 | |
| 		}
 | |
| 		return ($prefix ? '#' : '').$this->getModel()->getModelTable();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $prefix
 | |
| 	 * @param $quote
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getIfTableAlias($prefix = false, $quote = false) //XXX May no longer by useful. see joinAlias()
 | |
| 	{
 | |
| 		$alias = $this->getTableName(true);
 | |
| 		if($alias)
 | |
| 		{
 | |
| 			return $alias;
 | |
| 		}
 | |
| 		return ( !$quote ? $this->getTableName(false, $prefix) : '`'.$this->getTableName(false, $prefix).'`' );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get join table data - XXX DEPRECATE?
 | |
| 	 * @param string $table if null all data will be returned
 | |
| 	 * @param string $att_name search for specific attribute, default null (no search)
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function getJoinData($table = null, $att_name = null, $default_att = null)
 | |
| 	{		
 | |
| 		if($table === null)
 | |
| 		{
 | |
| 			return $this->tableJoin;
 | |
| 		}
 | |
| 		if($att_name === null)
 | |
| 		{
 | |
| 			return (isset($this->tableJoin[$table]) ? $this->tableJoin[$table] : array());
 | |
| 		}
 | |
| 		return (isset($this->tableJoin[$table][$att_name]) ? $this->tableJoin[$table][$att_name] : $default_att);
 | |
| 	}
 | |
| 
 | |
| 	public function setListOrder($order)
 | |
| 	{
 | |
| 		$this->listOrder = $order;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $table
 | |
| 	 * @param $data
 | |
| 	 * @return $this
 | |
| 	 */
 | |
| 	public function setJoinData($table, $data) //XXX - DEPRECATE?
 | |
| 	{
 | |
| 		if($data === null)
 | |
| 		{
 | |
| 			unset($this->tableJoin[$table]);
 | |
| 			return $this;
 | |
| 		}
 | |
| 		$this->tableJoin[$table] = (array) $data;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined model setter
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	protected function _setModel()
 | |
| 	{
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get current tree model
 | |
| 	 * @return e_admin_tree_model
 | |
| 	 */
 | |
| 	public function getTreeModel()
 | |
| 	{
 | |
| 		if($this->_tree_model === null)
 | |
| 		{
 | |
| 			$this->_setTreeModel();
 | |
| 		}
 | |
| 
 | |
| 		return $this->_tree_model;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get ordered models by their parents
 | |
| 	 * add extra
 | |
| 	 * @lonalore
 | |
| 	 * @return e_admin_tree_model
 | |
| 	 */
 | |
| 	public function getTreeModelSorted()
 | |
| 	{
 | |
| 		$tree = $this->getTreeModel();
 | |
| 
 | |
| 		$parentField = $this->getSortParent();
 | |
| 		$orderField = $this->getSortField();
 | |
| 
 | |
| 		$arr = array();
 | |
| 		/**
 | |
| 		 * @var  $id
 | |
| 		 * @var e_tree_model $model
 | |
| 		 */
 | |
| 		foreach ($tree->getTree() as $id => $model)
 | |
| 		{
 | |
| 			$parent = $model->get($parentField);
 | |
| 			$order = $model->get($orderField);
 | |
| 
 | |
| 			$model->set('_depth', '9999'); // include extra field in output, just as the MySQL function did.
 | |
| 
 | |
| 
 | |
| 			$arr[$id] = $model;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 	//	usort($arr); array_multisort() ?
 | |
| 
 | |
| 		$tree->setTree($arr,true); // set the newly ordered tree.
 | |
| 
 | |
| 	//	var_dump($arr);
 | |
| 
 | |
| 		return $this->_tree_model;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @lonalore - found online.
 | |
| 	 * @param string $idField       The item's ID identifier (required)
 | |
| 	 * @param string $parentField   The item's parent identifier (required)
 | |
| 	 * @param array $els            The array (required)
 | |
| 	 * @param int   $parentID       The parent ID for which to sort (internal)
 | |
| 	 * @param array $result         The result set (internal)
 | |
| 	 * @param int   $depth          The depth (internal)
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function parentChildSort_r($idField, $parentField, $els=array(), $parentID = 0, &$result = array(), &$depth = 0)
 | |
| 	{
 | |
| 	    foreach ($els as $key => $value)
 | |
| 	    {
 | |
| 	        if ($value[$parentField] == $parentID)
 | |
| 	        {
 | |
| 	            $value['depth'] = $depth;
 | |
| 	            array_push($result, $value);
 | |
| 	            unset($els[$key]);
 | |
| 	            $oldParent = $parentID;
 | |
| 	            $parentID = $value[$idField];
 | |
| 	            $depth++;
 | |
| 	            $this->parentChildSort_r($idField,$parentField, $els, $parentID, $result, $depth);
 | |
| 	            $parentID = $oldParent;
 | |
| 	            $depth--;
 | |
| 	        }
 | |
| 	    }
 | |
| 
 | |
| 	    return $result;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Set controller tree model
 | |
| 	 * @param e_admin_tree_model $tree_model
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	public function setTreeModel($tree_model)
 | |
| 	{
 | |
| 		$this->_tree_model = $tree_model;
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get currently parsed model while in list mode
 | |
| 	 * Model instance is registered by e_form::renderListForm()
 | |
| 	 *
 | |
| 	 * @return e_admin_model
 | |
| 	 */
 | |
| 	public function getListModel()
 | |
| 	{
 | |
| 		return e107::getRegistry('core/adminUI/currentListModel');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $model
 | |
| 	 * @return $this
 | |
| 	 */
 | |
| 	public function setListModel($model)
 | |
| 	{
 | |
| 		e107::setRegistry('core/adminUI/currentListModel', $model);
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined tree model setter
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	protected function _setTreeModel()
 | |
| 	{
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get extended (UI) Form instance
 | |
| 	 *
 | |
| 	 * @return e_admin_form_ui
 | |
| 	 */
 | |
| 	public function getUI()
 | |
| 	{
 | |
| 		if($this->_ui === null)
 | |
| 		{
 | |
| 			$this->_setUI();
 | |
| 		}
 | |
| 		return $this->_ui;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set controller UI form
 | |
| 	 * @param e_admin_form_ui $ui
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	public function setUI($ui)
 | |
| 	{
 | |
| 		$this->_ui = $ui;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined UI form setter
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	protected function _setUI()
 | |
| 	{
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get Config object
 | |
| 	 * @return e_plugin_pref or e_core_pref when used in core areas
 | |
| 	 */
 | |
| 	public function getConfig()
 | |
| 	{
 | |
| 		if($this->_pref === null)
 | |
| 		{
 | |
| 			$this->_setConfig();
 | |
| 		}
 | |
| 		return $this->_pref;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set Config object
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	public function setConfig($config)
 | |
| 	{
 | |
| 		$this->_prefs = $config; // XXX Discuss.
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $val
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	public function setBatchDelete($val)
 | |
| 	{
 | |
| 		$this->batchDelete = $val;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $val
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	public function setBatchCopy($val)
 | |
| 	{
 | |
| 		$this->batchCopy = $val;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined config setter
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	protected function _setConfig()
 | |
| 	{
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Manage column visibility
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	public function manageColumns()
 | |
| 	{
 | |
| 		$cols = array();
 | |
| 		$posted = $this->getPosted('e-columns', array());
 | |
| 		foreach ($this->getFields() as $field => $attr)
 | |
| 		{
 | |
| 			if((/*vartrue($attr['forced']) || */ in_array($field, $posted)) && !vartrue($attr['nolist']))
 | |
| 			{
 | |
| 				$cols[] = $field;
 | |
| 				continue;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Alow for an empty array to be saved also, to reset to default.
 | |
| 	    if($this->getPosted('etrigger_ecolumns', false)) // Column Save Button
 | |
| 		{
 | |
| 			$this->setUserPref($cols, $this->fieldPrefName);
 | |
| 			e107::getMessage()->addDebug('User Field Preferences Saved: ' .print_a($cols,true));
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Handle posted batch options routine
 | |
| 	 * @param string $batch_trigger
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	protected function _handleListBatch($batch_trigger)
 | |
| 	{
 | |
| 		$tp = e107::getParser();
 | |
| 		//$multi_name = vartrue($this->fields['checkboxes']['toggle'], 'multiselect');
 | |
| 		$multi_name = $this->getFieldAttr('checkboxes', 'toggle', 'multiselect');
 | |
| 		$selected = array_values($this->getPosted($multi_name, array()));
 | |
| 		$trigger = $tp->toDB(explode('__', $batch_trigger));
 | |
| 
 | |
| 		if(!empty($selected))
 | |
| 		{
 | |
| 			foreach ($selected as $i => $_sel) 
 | |
| 			{
 | |
| 				$selected[$i] = preg_replace('/[^\w\-:.]/', '', $_sel);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// XXX An empty selection should always be permitted for custom batch methods which may apply changes to all records, not only selected ones.
 | |
| 
 | |
| 
 | |
| 		if(strpos($batch_trigger, 'batch_') === 0)
 | |
| 		{
 | |
| 			list($tmp,$plugin,$command) = explode('_',$batch_trigger,3);
 | |
| 			$this->setPosted(array());
 | |
| 			$this->getRequest()->setAction('batch');
 | |
| 			$cls = e107::getAddon($plugin,'e_admin');
 | |
| 			e107::callMethod($cls,'process',$this,array('cmd'=>$command,'ids'=>$selected));
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		$this->setTriggersEnabled(false); //disable further triggering
 | |
| 
 | |
| 		$actionName = $this->getRequest()->getActionName();
 | |
| 
 | |
| 		if($actionName === 'Grid')
 | |
| 		{
 | |
| 			$actionName = 'List';
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		switch($trigger[0])
 | |
| 		{
 | |
| 
 | |
| 			case 'sefgen':
 | |
| 				$field = $trigger[1];
 | |
| 				$value = $trigger[2];
 | |
| 
 | |
| 				//handleListBatch(); for custom handling of all field names
 | |
| 				if(empty($selected))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				$method = 'handle'.$actionName.'SefgenBatch';
 | |
| 				if(method_exists($this, $method)) // callback handling
 | |
| 				{
 | |
| 					$this->$method($selected, $field, $value);
 | |
| 				}
 | |
| 			break;
 | |
| 
 | |
| 
 | |
| 			case 'export':
 | |
| 				if(empty($selected))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				$method = 'handle'.$actionName.'ExportBatch';
 | |
| 				if(method_exists($this, $method)) // callback handling
 | |
| 				{
 | |
| 					$this->$method($selected);
 | |
| 				}
 | |
| 
 | |
| 			break;
 | |
| 
 | |
| 			case 'delete':
 | |
| 				//method handleListDeleteBatch(); for custom handling of 'delete' batch
 | |
| 				// if(empty($selected)) return $this;
 | |
| 				// don't check selected data - subclass need to check additional post variables(confirm screen)
 | |
| 
 | |
| 				if(empty($selected) && !$this->getPosted('etrigger_delete_confirm')) // it's a delete batch, confirm screen
 | |
| 				{
 | |
| 					$params = $this->getFieldAttr($trigger[1], 'writeParms', array());
 | |
| 					if(!is_array($params))
 | |
| 					{
 | |
| 						parse_str($params, $params);
 | |
| 					}
 | |
| 					if(!vartrue($params['batchNoCheck']))
 | |
| 					{
 | |
| 						return $this;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				$method = 'handle'.$actionName.'DeleteBatch';
 | |
| 				if(method_exists($this, $method)) // callback handling
 | |
| 				{
 | |
| 					$this->$method($selected);
 | |
| 				}
 | |
| 			break;
 | |
| 
 | |
| 			case 'bool':
 | |
| 				if(empty($selected))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				$field = $trigger[1];
 | |
| 				$value = $trigger[2] ? 1 : 0;
 | |
| 				//something like handleListBoolBatch(); for custom handling of 'bool' batch
 | |
| 				$method = 'handle'.$actionName.'BoolBatch';
 | |
| 				if(method_exists($this, $method)) // callback handling
 | |
| 				{
 | |
| 					$this->$method($selected, $field, $value);
 | |
| 				}
 | |
| 			break;
 | |
| 
 | |
| 			case 'boolreverse':
 | |
| 				if(empty($selected))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				$field = $trigger[1];
 | |
| 				//something like handleListBoolreverseBatch(); for custom handling of 'boolreverse' batch
 | |
| 				$method = 'handle'.$actionName.'BoolreverseBatch';
 | |
| 				if(method_exists($this, $method)) // callback handling
 | |
| 				{
 | |
| 					$this->$method($selected, $field);
 | |
| 				}
 | |
| 			break;
 | |
| 			
 | |
| 			// see commma, userclasses batch options
 | |
| 			case 'attach':
 | |
| 			case 'deattach':
 | |
| 			case 'addAll':
 | |
| 			case 'clearAll':
 | |
| 				if(empty($selected))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				$field = $trigger[1];
 | |
| 				$value = $trigger[2];
 | |
| 				
 | |
| 				if($trigger[0] === 'addAll')
 | |
| 				{
 | |
| 					$parms = $this->getFieldAttr($field, 'writeParms', array());
 | |
| 					if(!is_array($parms))
 | |
| 					{
 | |
| 						parse_str($parms, $parms);
 | |
| 					}
 | |
| 					unset($parms['__options']);
 | |
| 					$value = $parms;
 | |
| 					if(empty($value))
 | |
| 					{
 | |
| 						return $this;
 | |
| 					}
 | |
| 					if(!is_array($value))
 | |
| 					{
 | |
| 						$value = array_map('trim', explode(',', $value));
 | |
| 					}
 | |
| 				}
 | |
| 				
 | |
| 				if(method_exists($this, 'handleCommaBatch')) 
 | |
| 				{
 | |
| 					$this->handleCommaBatch($selected, $field, $value, $trigger[0]);
 | |
| 				}
 | |
| 			break;
 | |
| 			
 | |
| 			// append to userclass list
 | |
| 			case 'ucadd':
 | |
| 			case 'ucremove':
 | |
| 				if(empty($selected))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				$field = $trigger[1];
 | |
| 				$class = $trigger[2];
 | |
| 				$user = e107::getUser();
 | |
| 				$e_userclass = e107::getUserClass(); 
 | |
| 				
 | |
| 				// check userclass manager class
 | |
| 				if (!isset($e_userclass->class_tree[$class]) || !$user->checkClass($e_userclass->class_tree[$class]))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 
 | |
| 				if(method_exists($this, 'handleCommaBatch')) 
 | |
| 				{
 | |
| 					$trigger[0] = $trigger[0] === 'ucadd' ? 'attach' : 'deattach';
 | |
| 					$this->handleCommaBatch($selected, $field, $class, $trigger[0]);
 | |
| 				}
 | |
| 			break;
 | |
| 			
 | |
| 			// add all to userclass list
 | |
| 			// clear userclass list
 | |
| 			case 'ucaddall':
 | |
| 			case 'ucdelall':
 | |
| 				if(empty($selected))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				$field = $trigger[1];
 | |
| 				$user = e107::getUser();
 | |
| 				$e_userclass = e107::getUserClass(); 
 | |
| 				$parms = $this->getFieldAttr($field, 'writeParms', array());
 | |
| 				if(!is_array($parms))
 | |
| 				{
 | |
| 					parse_str($parms, $parms);
 | |
| 				}
 | |
| 				if(!vartrue($parms['classlist']))
 | |
| 				{
 | |
| 					return $this;
 | |
| 				}
 | |
| 				
 | |
| 				$classes = $e_userclass->uc_required_class_list($parms['classlist']);
 | |
| 				foreach ($classes as $id => $label) 
 | |
| 				{
 | |
| 					// check userclass manager class
 | |
| 					if (!isset($e_userclass->class_tree[$id]) || !$user->checkClass($e_userclass->class_tree[$id]))
 | |
| 					{
 | |
| 						$msg = $tp->lanVars(LAN_NO_ADMIN_PERMISSION,$label);
 | |
| 						$this->getTreeModel()->addMessageWarning($msg);
 | |
| 						unset($classes[$id],$msg);
 | |
| 					}
 | |
| 				}
 | |
| 				if(method_exists($this, 'handleCommaBatch'))
 | |
| 				{
 | |
| 					$this->handleCommaBatch($selected, $field, array_keys($classes), $trigger[0] === 'ucdelall' ? 'clearAll' : 'addAll');
 | |
| 				}
 | |
| 			break;
 | |
| 
 | |
| 			// handleListCopyBatch etc.
 | |
| 			default:
 | |
| 				$field = $trigger[0];
 | |
| 				$value = $trigger[1];
 | |
| 
 | |
| 				//something like handleListUrlTypeBatch(); for custom handling of 'url_type' field name
 | |
| 				$method = 'handle'.$actionName.$this->getRequest()->camelize($field).'Batch';
 | |
| 
 | |
| 				e107::getMessage()->addDebug('Searching for custom batch method: ' .$method. '(' .$selected. ',' .$value. ')');
 | |
| 
 | |
| 				if(method_exists($this, $method)) // callback handling
 | |
| 				{
 | |
| 					$this->$method($selected, $value);
 | |
| 					break;
 | |
| 				}
 | |
| 
 | |
| 				//handleListBatch(); for custom handling of all field names
 | |
| 				//if(empty($selected)) return $this;
 | |
| 				$method = 'handle'.$actionName.'Batch';
 | |
| 				e107::getDebug()->log('Checking for batch method: ' .$method);
 | |
| 				if(method_exists($this, $method))
 | |
| 				{
 | |
| 					$this->$method($selected, $field, $value);
 | |
| 				}
 | |
| 
 | |
| 
 | |
| 
 | |
| 			break;
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Handle requested filter dropdown value
 | |
| 	 * @param string $filter_value
 | |
| 	 * @return array field -> value
 | |
| 	 */
 | |
| 	protected function _parseFilterRequest($filter_value)
 | |
| 	{
 | |
| 		$tp = e107::getParser();
 | |
| 		if(!$filter_value || $filter_value === '___reset___')
 | |
| 		{
 | |
| 			return array();
 | |
| 		}
 | |
| 		$filter = (array) $tp->toDB(explode('__', $filter_value));
 | |
| 		$res = array();
 | |
| 		switch($filter[0])
 | |
| 		{
 | |
| 			case 'bool':
 | |
| 				// direct query
 | |
| 				$res = array($filter[1], $filter[2]);
 | |
| 				$this->_log('listQry Filtered by ' .$filter[1]. ' (' .($filter[2] ? 'true': 'false'). ')');
 | |
| 			break;
 | |
| 			
 | |
| 			case 'datestamp':
 | |
| 							
 | |
| 				//XXX DO NOT TRANSLATE THESE VALUES!
 | |
| 				$dateConvert = array(
 | |
| 					'hour'    => '1 hour ago',
 | |
| 					'day'     => '24 hours ago',
 | |
| 					'week'    => '1 week ago',
 | |
| 					'today'     => 'today midnight',
 | |
| 					'thisweek' => 'monday this week midnight',
 | |
| 					'thismonth' => 'first day of this month midnight',
 | |
| 					'thisyear' => 'first day of January midnight',
 | |
| 					'month'   => '1 month ago',
 | |
| 					'month3'  => '3 months ago',
 | |
| 					'month6'  => '6 months ago',
 | |
| 					'month9'  => '9 months ago',
 | |
| 					'year'    => '1 year ago',
 | |
| 					'nhour'   => 'now + 1 hour',
 | |
| 					'nday'    => 'now + 24 hours',
 | |
| 					'nweek'   => 'now + 1 week',
 | |
| 					'nmonth'  => 'now + 1 month',
 | |
| 					'nmonth3' => 'now + 3 months',
 | |
| 					'nmonth6' => 'now + 6 months',
 | |
| 					'nmonth9' => 'now + 9 months',
 | |
| 					'nyear'   => 'now + 1 year',
 | |
| 				);
 | |
| 				
 | |
| 				$ky = $filter[2];
 | |
| 				$time = vartrue($dateConvert[$ky]);
 | |
| 				$timeStamp = strtotime($time);
 | |
| 
 | |
| 				$res = array($filter[1], $timeStamp);
 | |
| 			//	e107::getMessage()->addDebug('Date: '.date('c', $timeStamp));
 | |
| 				$this->_log('listQry Filtered by ' .$filter[1]. ' (' .$time. ')');
 | |
| 				
 | |
| 			break;
 | |
| 
 | |
| 			default:
 | |
| 				//something like handleListUrlTypeFilter(); for custom handling of 'url_type' field name filters
 | |
| 				$method = 'handle'.$this->getRequest()->getActionName().$this->getRequest()->camelize($filter[0]).'Filter';
 | |
| 				$args = array_slice($filter, 1);
 | |
| 
 | |
| 				e107::getMessage()->addDebug('Searching for custom filter method: ' .$method. '(' .implode(', ', $args). ')');
 | |
| 
 | |
| 				if (property_exists($this, $method)) // dynamic property handling
 | |
| 				{
 | |
| 					e107::getMessage()->addDebug('Accessing filter property <strong>' . get_class($this) . '::$' . $method . '</strong>');
 | |
| 
 | |
| 					if (is_callable($this->$method))
 | |
| 					{
 | |
| 						e107::getMessage()->addDebug('Executing callable property <strong>' . get_class($this) . '::$' . $method . '(' . implode(', ', $args) . ')</strong>');
 | |
| 
 | |
| 						return call_user_func_array($this->$method, $args);
 | |
| 					}
 | |
| 
 | |
| 					// If not callable, return the property value as is (could be a query fragment or other data)
 | |
| 					e107::getMessage()->addDebug('Returning property value from <strong>' . get_class($this) . '::$' . $method . '</strong>');
 | |
| 
 | |
| 					return $this->$method;
 | |
| 				}
 | |
| 				elseif (method_exists($this, $method)) // method handling
 | |
| 				{
 | |
| 					e107::getMessage()->addDebug('Executing filter callback <strong>' . get_class($this) . '::' . $method . '(' . implode(', ', $args) . ')</strong>');
 | |
| 
 | |
| 					return call_user_func_array([$this, $method], $args);
 | |
| 				}
 | |
| 
 | |
| 				$res = array($filter[0], $filter[1]);
 | |
| 				$this->_log('listQry Filtered by ' .$filter[0]. ' (' .$filter[1]. ')');
 | |
| 				break;
 | |
| 		}
 | |
| 
 | |
| 		//print_a($res);
 | |
| 		//exit;
 | |
| 
 | |
| 		return $res;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Convert posted to model values after submit (based on field type)
 | |
| 	 * @param array $data
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function convertToData(&$data)
 | |
| 	{
 | |
| 		$model = new e_model($data);
 | |
| 	
 | |
| 		foreach ($this->getFields() as $key => $attributes)
 | |
| 		{
 | |
| 			$value = vartrue($attributes['dataPath']) ? $model->getData($attributes['dataPath'])  : $model->get($key);
 | |
| 
 | |
| 			if($value === null)
 | |
| 			{
 | |
| 				continue;
 | |
| 			}
 | |
| 			switch($attributes['type'])
 | |
| 			{
 | |
| 			
 | |
| 				case 'password': //TODO more encryption options. 
 | |
| 					if(strlen($value) < 30) // expect a non-md5 value if less than 32 chars. 
 | |
| 					{
 | |
| 						$value = md5($value);
 | |
| 					}
 | |
| 					
 | |
| 				break;	
 | |
| 			
 | |
| 			
 | |
| 				case 'datestamp':
 | |
| 					$opt = array();
 | |
| 					if(!is_numeric($value))
 | |
| 					{
 | |
| 						if(!empty($attributes['writeParms']))
 | |
| 						{
 | |
| 							if(is_string($attributes['writeParms']))
 | |
| 							{
 | |
| 								parse_str($attributes['writeParms'],$opt);
 | |
| 							}
 | |
| 							elseif(is_array($attributes['writeParms']))
 | |
| 							{
 | |
| 								$opt = $attributes['writeParms'];
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 						
 | |
| 						$format = !empty($opt['type']) ? ('input'.$opt['type']) : 'inputdate';
 | |
| 
 | |
| 						if($attributes['data'] !== false)
 | |
| 						{
 | |
| 							$value = trim($value) ? e107::getDate()->toTime($value, $format) : 0;
 | |
| 						}
 | |
| 					}
 | |
| 				break;
 | |
| 
 | |
| 				case 'ip': // TODO - ask Steve if this check is required
 | |
| 					//if(strpos($value, '.') !== FALSE)
 | |
| 					{
 | |
| 						$value = trim($value) ? e107::getIPHandler()->ipEncode($value) : '';
 | |
| 					}
 | |
| 				break;
 | |
| 
 | |
| 				case 'dropdown': // TODO - ask Steve if this check is required
 | |
| 				case 'lanlist':
 | |
| 				case 'userclasses':
 | |
| 				case 'comma':
 | |
| 				case 'checkboxes':
 | |
| 					if(is_array($value))
 | |
| 					{
 | |
| 						// no sanitize here - data is added to model posted stack
 | |
| 						// and validated & sanitized before sent to db
 | |
| 						//$value = array_map(array(e107::getParser(), 'toDB'), $value);
 | |
| 						$value = implode(',', $value);
 | |
| 					}
 | |
| 				break;
 | |
| 				
 | |
| 				case 'images':
 | |
| 				case 'files':
 | |
| 		
 | |
| 				//	XXX Cam @ SecretR: didn't work here. See model_class.php line 2046. 
 | |
| 				// if(!is_array($value))
 | |
| 			//		{
 | |
| 				//		$value = e107::unserialize($value);	
 | |
| 				//	}
 | |
| 				break;
 | |
| 				
 | |
| 	
 | |
| 			}
 | |
| /*
 | |
| 			if($attributes['serialize'] == true)
 | |
| 			{
 | |
| 				$attributes['data'] = 'array';		
 | |
| 			}
 | |
| 
 | |
| 			if($attributes['data'] != 'array')
 | |
| 			{
 | |
| 				$value = e107::unserialize($value);	
 | |
| 			}
 | |
| */
 | |
| 	
 | |
| 			if(!empty($attributes['dataPath']))
 | |
| 			{
 | |
| 				$model->setData($attributes['dataPath'], $value);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$model->set($key, $value);
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		$data = $model->getData();
 | |
| 		unset($model);
 | |
| 		$this->toData($data);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined method for converting POSTED to MODEL data
 | |
| 	 * @param array $data posted data
 | |
| 	 * @param string $type current action type - edit, create, list or user defined
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function toData(&$data, $type = '')
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Take approproate action after successfull submit
 | |
| 	 *
 | |
| 	 * @param integer $id optional, needed only if redirect action is 'edit'
 | |
| 	 * @param string $noredirect_for don't redirect if action equals to its value
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	protected function doAfterSubmit($id = 0, $noredirect_for = '')
 | |
| 	{
 | |
| 		if(e_AJAX_REQUEST)
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 		
 | |
| 		if($noredirect_for && $noredirect_for == $this->getPosted('__after_submit_action') && $noredirect_for == $this->getAction())
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		$choice = $this->getPosted('__after_submit_action', 0);
 | |
| 		switch ($choice) {
 | |
| 			case 'create': // create
 | |
| 				$this->redirectAction('create', 'id');
 | |
| 			break;
 | |
| 
 | |
| 			case 'edit': // edit
 | |
| 				$this->redirectAction('edit', '', 'id='.$id);
 | |
| 			break;
 | |
| 
 | |
| 			case 'list': // list
 | |
| 				$this->redirectAction('list', 'id');
 | |
| 			break;
 | |
| 
 | |
| 			default:
 | |
| 				$choice = explode('|', str_replace('{ID}', $id, $choice), 3);
 | |
| 				$this->redirectAction(preg_replace('/[^\w\-:.]/', '', $choice[0]), vartrue($choice[1]), vartrue($choice[2]));
 | |
| 			break;
 | |
| 		}
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Build ajax auto-complete filter response
 | |
| 	 * @return string response markup
 | |
| 	 */
 | |
| 	protected function renderAjaxFilterResponse($listQry = '')
 | |
| 	{
 | |
| 		$debug = false;
 | |
| 		$srch = $this->getPosted('searchquery');
 | |
| 		$this->getRequest()->setQuery('searchquery', $srch); //_modifyListQry() is requiring GET String
 | |
| 
 | |
| 		$ret = '<ul>';
 | |
| 		$ret .= '<li>'.$srch.'<span class="informal warning"> '.LAN_FILTER_LABEL_TYPED.'</span></li>'; // fix Enter - search for typed word only
 | |
| 
 | |
| 		$reswords = array();
 | |
| 		if(trim($srch) !== '')
 | |
| 		{
 | |
| 			// Build query
 | |
| 			$qry = $this->_modifyListQry(false, true, 0, 20, $listQry);
 | |
| 			$this->_log('Filter ListQry: ' .$qry);
 | |
| 			//file_put_contents(e_LOG.'uiAjaxResponseSQL.log', $qry."\n\n", FILE_APPEND);
 | |
| 
 | |
| 			// Make query
 | |
| 			$sql = e107::getDb();
 | |
| 			if($qry && $sql->gen($qry, $debug))
 | |
| 			{
 | |
| 				while ($res = $sql->fetch())
 | |
| 				{
 | |
| 					$tmp1 = array();
 | |
| 					$tmp = array_values(preg_grep('#'.$srch.'#i', $res));
 | |
| 					foreach ($tmp as $w)
 | |
| 					{
 | |
| 						if($w == $srch)
 | |
| 						{
 | |
| 							array_unshift($reswords, $w); //exact match
 | |
| 							continue;
 | |
| 						}
 | |
| 						preg_match('#[\S]*('.$srch.')[\S]*#i', $w, $tmp1);
 | |
| 						if($tmp1[0])
 | |
| 						{
 | |
| 							$reswords[] = $tmp1[0];
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// Build response
 | |
| 			$reswords = array_unique($reswords);
 | |
| 			if($reswords)
 | |
| 			{
 | |
| 				$ret .= '<li>'.implode("</li>\n\t<li>", $reswords).'</li>';
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$ret .= '<li><span class="informal warning"> '.LAN_FILTER_LABEL_CLEAR.' </span></li>'; // clear filter option
 | |
| 		$ret .= '</ul>';
 | |
| 		return $ret;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Given an alias such as 'u' or 'n.news_datestamp' -  will return the associated table such as 'user' or 'news'
 | |
| 	 */
 | |
| 	public function getTableFromAlias($alias)
 | |
| 	{
 | |
| 		if(strpos($alias, '.')!==false)
 | |
| 		{
 | |
| 			list($alias,$tmp) = explode('.',$alias,2);
 | |
| 		}
 | |
| 				
 | |
| 		$tmp = array_flip($this->joinAlias);
 | |
| 		return vartrue($tmp[$alias]);			
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $field
 | |
| 	 * @return array|false|mixed
 | |
| 	 */
 | |
| 	public function getJoinField($field=null)
 | |
| 	{
 | |
| 		if(empty($field))
 | |
| 		{
 | |
| 			return $this->joinField;
 | |
| 		}
 | |
| 
 | |
| 		return isset($this->joinField[$field]) ? $this->joinField[$field] : false; // vartrue($this->joinField[$field],false);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getJoinAlias()
 | |
| 	{
 | |
| 		return $this->joinAlias;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Parses all available field data, adds internal attributes for handling join requests
 | |
| 	 * @return e_admin_controller_ui
 | |
| 	 */
 | |
| 	protected function parseAliases()
 | |
| 	{
 | |
| 		if($this->_alias_parsed)
 | |
| 		{
 | |
| 			return $this;
 | |
| 		} // already parsed!!!
 | |
| 
 | |
| 		$this->joinAlias($this->listQry); // generate Table Aliases from listQry
 | |
| 		
 | |
| 		if($this->getJoinData())
 | |
| 		{
 | |
| 			foreach ($this->getJoinData() as $table => $att)
 | |
| 			{
 | |
| 				if(strpos($table, '.') !== false)
 | |
| 				{
 | |
| 					$tmp = explode('.', $table, 2);
 | |
| 					$this->setJoinData($table, null);
 | |
| 					$att['alias'] = $tmp[0];
 | |
| 					$att['table'] = $tmp[1];
 | |
| 					$att['__tablePath'] = $att['alias'].'.';
 | |
| 					$att['__tableFrom'] = '`#'.$att['table'].'` AS '.$att['alias'];
 | |
| 					$this->setJoinData($att['alias'], $att);
 | |
| 					unset($tmp);
 | |
| 					continue;
 | |
| 				}
 | |
| 				$att['table'] = $table;
 | |
| 				$att['alias'] = '';
 | |
| 				$att['__tablePath'] = '`#'.$att['table'].'`.';
 | |
| 				$att['__tableFrom'] = '`#'.$att['table'].'`';
 | |
| 				$this->setJoinData($table, $att);
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 
 | |
| 		if(empty($this->fields))
 | |
| 		{
 | |
| 			$this->_alias_parsed = true;
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		// check for table & field aliases
 | |
| 		$fields = array(); // preserve order
 | |
| 		foreach ($this->fields as $field => $att)
 | |
| 		{
 | |
| 			// fieldAlias.fieldName // table name no longer required as it's included in listQry. (see joinAlias() )
 | |
| 			if(strpos($field, '.') !== false) // manually entered alias.
 | |
| 			{
 | |
| 				$tmp = explode('.', $field, 2);
 | |
| 				$table = $this->getTableFromAlias($tmp[0]);
 | |
| 				$att['table'] = $table;
 | |
| 				$att['alias'] = $tmp[0];
 | |
| 				$att['field'] = $tmp[1];
 | |
| 				$att['__tableField'] = $field;
 | |
| 				$att['__tablePath'] = $att['alias'].'.';
 | |
| 				$att['__tableFrom'] = '`#' .$table. '`.' .$tmp[1];//." AS ".$att['alias'];
 | |
| 				$field = $att['alias'] ? $tmp[1] : $tmp[0];
 | |
| 
 | |
| 				$fields[$field] = $att;
 | |
| 				unset($tmp);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 
 | |
| 				$att['table'] = $this->getIfTableAlias();
 | |
| 				
 | |
| 				if($newField = $this->getJoinField($field)) // Auto-Detect. 
 | |
| 				{
 | |
| 					$table = $this->getTableFromAlias($newField); // Auto-Detect. 
 | |
| 					$att['table'] = $table;
 | |
| 					$att['alias'] = $newField;
 | |
| 					$att['__tableField'] = $newField;	
 | |
| 					// $att['__tablePath'] = $newField; ????!!!!!
 | |
| 					$att['__tableFrom'] = '`#' .$table. '`.' .$field;//." AS ".$newField;
 | |
| 				}			
 | |
| 				elseif(isset($this->joinAlias[$this->table]) && $field !== 'checkboxes' && $field !== 'options')
 | |
| 				{
 | |
| 					$att['alias'] = $this->joinAlias[$this->table]. '.' .$field;
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					$att['alias'] = '';
 | |
| 				}
 | |
| 				$att['field'] = $field;
 | |
| 				$fields[$field] = $att;
 | |
| 			}
 | |
| 
 | |
| 			if($fields[$field]['table'] == $this->getIfTableAlias())
 | |
| 			{
 | |
| 				$fields[$field]['__tableField'] = $att['alias'] ? $att['alias'] : $this->getIfTableAlias(true, true).'.'.$att['field'];
 | |
| 				$fields[$field]['__tableFrom'] = $this->getIfTableAlias(true, true).'.'.$att['field'].($att['alias'] ? ' AS '.$att['alias'] : '');
 | |
| 			}
 | |
| 		//	else
 | |
| 			{
 | |
| 		//		$fields[$field]['__tableField'] = $this->getJoinData($fields[$field]['table'], '__tablePath').$field;
 | |
| 			}
 | |
| 			/*
 | |
| 			if($fields[$field]['table'])
 | |
| 			{
 | |
| 				if($fields[$field]['table'] == $this->getIfTableAlias(false))
 | |
| 				{
 | |
| 					$fields[$field]['__tableField'] = $att['alias'] ? $att['alias'] : $this->getIfTableAlias(true, true).'.'.$att['field'];
 | |
| 					$fields[$field]['__tableFrom'] = $this->getIfTableAlias(true, true).'.'.$att['field'].($att['alias'] ? ' AS '.$att['alias'] : '');
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					$fields[$field]['__tableField'] = $this->getJoinData($fields[$field]['table'], '__tablePath').$field;
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$fields[$field]['__tableField'] = '`'.$this->getTableName(false, true).'`.'.$field;
 | |
| 			}
 | |
| 			*/
 | |
| 		}
 | |
| 
 | |
| 	
 | |
| 		$this->fields = $fields;
 | |
| 
 | |
| 		$this->_alias_parsed = true;
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 *  Intuitive LEFT JOIN Qry support. (preferred)
 | |
| 	 *  Generate array of table names and their alias - auto-detected from listQry;
 | |
| 	 *  eg. $listQry = "SELECT m.*, u.user_id,u.user_name FROM #core_media AS m LEFT JOIN #user AS u ON m.media_author = u.user_id";
 | |
| 	 */
 | |
| 	public function joinAlias($listQry=null)
 | |
| 	{
 | |
| 		if(!empty($listQry))
 | |
| 		{
 | |
| 			preg_match_all("/`?#([\w-]+)`?\s*(as|AS)\s*([\w-]+)/im",$listQry,$matches);
 | |
| 			$keys = array();
 | |
| 			foreach($matches[1] AS $k=>$v)
 | |
| 			{
 | |
| 				if(!array_key_exists($v, $this->joinAlias) && !empty($matches[3][$k]))
 | |
| 				{
 | |
| 					$this->joinAlias[$v] = $matches[3][$k]; // array. eg $this->joinAlias['core_media'] = 'm';
 | |
| 				}
 | |
| 				
 | |
| 				$keys[] = $matches[3][$k];
 | |
| 			}
 | |
| 
 | |
| 			foreach($keys as $alias)
 | |
| 			{
 | |
| 				preg_match_all('/' .$alias."\.([\w]*)/i",$listQry,$match);
 | |
| 				foreach($match[1] as $k=>$m)
 | |
| 				{
 | |
| 					if(empty($m))
 | |
| 					{
 | |
| 						continue;
 | |
| 					}
 | |
| 					$this->joinField[$m] = $match[0][$k];		
 | |
| 				}					
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 		elseif($this->tableJoin)
 | |
| 		{
 | |
| 			foreach ($this->tableJoin as $tbl => $data) 
 | |
| 			{
 | |
| 				$matches = explode('.', $tbl, 2);
 | |
| 				$this->joinAlias[$matches[1]] = $matches[0]; // array. eg $this->joinAlias['core_media'] = 'm';
 | |
| 				//'user_name'=>'u.user_name'
 | |
| 				if(isset($data['fields']) && $data['fields'] !== '*')
 | |
| 				{
 | |
| 					$tmp = explode(',', $data['fields']);
 | |
| 					foreach ($tmp as $field) 
 | |
| 					{
 | |
| 						$this->joinField[$field] = $matches[0].'.'.$field;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Quick fix for bad custom $listQry; 
 | |
| 	 */
 | |
| 	protected function parseCustomListQry($qry)
 | |
| 	{
 | |
| 		if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 		{
 | |
| 			e107::getMessage()->addDebug('Using Custom listQry ');	
 | |
| 		}
 | |
| 			
 | |
| 		if(strpos($qry,'`')===false && strpos($qry, 'JOIN')===false) 
 | |
| 		{
 | |
| 			$ret = preg_replace("/FROM\s*(#[\w]*)/", 'FROM `$1`', $qry);  // backticks missing, so add them.
 | |
| 						
 | |
| 			if($ret)
 | |
| 			{
 | |
| 				e107::getMessage()->addDebug('Your $listQry is missing `backticks` around the table name! It should look like this'. print_a($ret,true)); 
 | |
| 				return $ret; 	
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		return $qry; 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Fix search string by replacing the commonly used '*' wildcard
 | |
| 	 * with the mysql represenation of it '%' and '?' with '_' (single character)
 | |
| 	 *
 | |
| 	 * @param string $search
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	protected function fixSearchWildcards($search)
 | |
| 	{
 | |
| 		$search = trim($search);
 | |
| 		if (empty($search))
 | |
| 		{
 | |
| 			return '';
 | |
| 		}
 | |
| 
 | |
| 		// strip wildcard on the beginning and the end
 | |
| 		while (strpos($search, '*') === 0)
 | |
| 		{
 | |
| 			$search = substr($search, 1);
 | |
| 		}
 | |
| 		while (substr($search, -1) === '*')
 | |
| 		{
 | |
| 			$search = substr($search, 0, -1);
 | |
| 		}
 | |
| 
 | |
| 		if(strpos($search,'"')===0 || strpos($search,''')===0)
 | |
| 		{
 | |
| 			$search = str_replace(['"','''],'',$search);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$search = str_replace(' ','|',$search);
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		// replace "*" wildcard with mysql wildcard "%"
 | |
| 		return str_replace(array('*', '?'), array('%', '_'), $search);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	// TODO - abstract, array return type, move to parent?
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $raw
 | |
| 	 * @param $isfilter
 | |
| 	 * @param $forceFrom
 | |
| 	 * @param $forceTo
 | |
| 	 * @param $listQry
 | |
| 	 * @return array|Custom|false|string|string[]
 | |
| 	 */
 | |
| 	protected function _modifyListQry($raw = false, $isfilter = false, $forceFrom = false, $forceTo = false, $listQry = '')
 | |
| 	{
 | |
| 		
 | |
| 		$request    = $this->getRequest();
 | |
| 
 | |
| 		
 | |
| 		$tablePath  = $this->getIfTableAlias(true, true).'.';
 | |
| 		$tableFrom  = '`'.$this->getTableName(false, true).'`'.($this->getTableName(true) ? ' AS '.$this->getTableName(true) : '');
 | |
| 		$primaryName = $this->getPrimaryName();
 | |
| 		$perPage    = (int) $this->getPerPage();
 | |
| 				
 | |
| 		$qryField   = $request->getQuery('field');
 | |
| 		$qryAsc     = $request->getQuery('asc');
 | |
| 		$qryFrom    = (int) $request->getQuery('from', 0);
 | |
| 		$orderField = $request->getQuery('field', $this->getDefaultOrderField());
 | |
| 		$filterOptions = $request->getQuery('filter_options', '');
 | |
| 		$searchTerm     =$request->getQuery('searchquery', '');
 | |
| 		$handleAction = $this->getRequest()->getActionName();
 | |
| 
 | |
| 		return $this->_modifyListQrySearch($listQry, $searchTerm, $filterOptions, $tablePath, $tableFrom, $primaryName, $raw, $orderField, $qryAsc, $forceFrom, $qryFrom, $forceTo, $perPage, $qryField,  $isfilter, $handleAction);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Return a Parent/Child SQL Query based on sortParent and sortField variables
 | |
| 	 *
 | |
| 	 * Note: Since 2018-01-28, the queries were replaced with pure PHP sorting. See:
 | |
| 	 *       https://github.com/e107inc/e107/issues/3015
 | |
| 	 *
 | |
| 	 * @param bool|false $orderby - include 'ORDER BY' in the qry.
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getParentChildQry($orderby=false)
 | |
| 	{
 | |
| 		return 'SELECT * FROM `#' .$this->getTableName(). '` ';
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Creates a backup record in the admin_history table for a given action on a specific record.
 | |
| 	 *
 | |
| 	 * @param string $table  The name of the table where the record resides.
 | |
| 	 * @param int    $pid    The primary ID field of the record.
 | |
| 	 * @param int    $id     The ID of the specific record being backed up.
 | |
| 	 * @param string $action The action performed on the record (e.g., 'update' or 'delete').
 | |
| 	 * @param array  $data   An associative array of field data to be included in the history record.
 | |
| 	 * @param bool  $posted Whether the data has been posted and requires additional filter based on current $fields values or not.
 | |
| 	 * @return bool True on successful creation of the backup record, false on failure.
 | |
| 	 */
 | |
| 	protected function backupToHistory($table, $pid, $id, $action, $data, $posted = true)
 | |
| 	{
 | |
| 		if($posted)
 | |
| 		{
 | |
| 			foreach($data as $field=>$var)
 | |
| 			{
 | |
| 				if(empty($this->fields[$field]['data'])) // exclude data not in the table.
 | |
| 				{
 | |
| 					unset($data[$field]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$historyData = [
 | |
| 			'history_table'     => $table,
 | |
| 			'history_pid'       => $pid,
 | |
| 			'history_record_id' => $id,
 | |
| 			'history_action'    => $action, // 'update' or 'delete'
 | |
| 			'history_data'      => json_encode($data, JSON_PRETTY_PRINT),
 | |
| 			'history_user_id'   => USERID,
 | |
| 			'history_datestamp' => time(),
 | |
| 		];
 | |
| 
 | |
| 		// Insert the record into the admin_history table
 | |
| 		if (!e107::getDb()->insert('admin_history', $historyData))
 | |
| 		{
 | |
| 			e107::getMessage()->addError("Failed to save history for table '{$table}', record ID {$id}");
 | |
| 			e107::getMessage()->addError(e107::getDb()->getLastErrorText());
 | |
| 			e107::getMessage()->addError(print_a($historyData, true));
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		// Optional: Add debug logs for successful history creation
 | |
| 		e107::getMessage()->addDebug("History saved for table '{$table}', record ID {$id}");
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Manage submit item
 | |
| 	 * Note: $callbackBefore will break submission if returns false
 | |
| 	 *
 | |
| 	 * @param string $callbackBefore existing method from $this scope to be called before submit
 | |
| 	 * @param string $callbackAfter existing method from $this scope to be called after successfull submit
 | |
| 	 * @param string $noredirectAction passed to doAfterSubmit()
 | |
| 	 * @return boolean
 | |
| 	 */
 | |
| 	protected function _manageSubmit($callbackBefore = '', $callbackAfter = '', $callbackError = '', $noredirectAction = '', $forceSave=false)
 | |
| 	{
 | |
| 
 | |
| 		$model = $this->getModel();
 | |
| 		$old_data = $model->getData();
 | |
| 
 | |
| 		$_posted = $this->getPosted();
 | |
| 		$this->convertToData($_posted);
 | |
| 
 | |
| 		if($callbackBefore && method_exists($this, $callbackBefore))
 | |
| 		{
 | |
| 			$data = $this->$callbackBefore($_posted, $old_data, $model->getId());
 | |
| 			if($data === false)
 | |
| 			{
 | |
| 				// we don't wanna loose posted data
 | |
| 				$model->setPostedData($_posted);
 | |
| 				return false;
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 
 | |
| 			if($data && is_array($data))
 | |
| 			{
 | |
| 				// add to model data fields array if required
 | |
| 				foreach ($data as $f => $val)
 | |
| 				{
 | |
| 					if($this->getFieldAttr($f, 'data'))
 | |
| 					{
 | |
| 						$model->setDataField($f, $this->getFieldAttr($f, 'data'));
 | |
| 					}
 | |
| 				}
 | |
| 				$_posted = array_merge($_posted, $data);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 	//	$model->addMessageDebug(print_a($_posted,true));
 | |
| 	//	$model->addMessageDebug(print_a($this,true));
 | |
| 
 | |
| 		// - Autoincrement sortField on 'Create'.
 | |
| 
 | |
| 
 | |
| 		// Prevent parent being assigned as self.
 | |
| 		if(!empty($this->sortParent) && $this->getAction() === 'edit' && ($model->getId() == $_posted[$this->sortParent] ) )
 | |
| 		{
 | |
| 			$vars = array(
 | |
| 				'x'=> $this->getFieldAttr($this->sortParent,'title'),
 | |
| 				'y'=> $this->getFieldAttr($this->pid,'title'),
 | |
| 			);
 | |
| 
 | |
| 			$message = e107::getParser()->lanVars(LAN_UI_X_CANT_EQUAL_Y, $vars);
 | |
| 			$model->addMessageWarning($message);
 | |
| 			$model->setMessages();
 | |
| 			$this->getUI()->addWarning($this->sortParent);
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 		if( !empty($this->sortField) && empty($this->sortParent) && empty($_posted[$this->sortField]) && ($this->getAction() === 'create'))
 | |
| 		{
 | |
| 
 | |
| 			$incVal = e107::getDb()->max($this->table, $this->sortField) + 1;
 | |
| 			$_posted[$this->sortField] = $incVal;
 | |
| 		//	$model->addMessageInfo(print_a($_posted,true));
 | |
| 		}
 | |
| 
 | |
| 		$id = $model->getId();
 | |
| 
 | |
| 		// Trigger Plugin Admin-ui event.  'pre'
 | |
| 		if($triggerName = $this->getEventTriggerName($this->getEventName(), $_posted['etrigger_submit'])) // 'create' or 'update';
 | |
| 		{
 | |
| 			if($halt = $this->triggerEvent($triggerName, $_posted,$old_data,$id))
 | |
| 			{
 | |
| 				$model->setMessages();
 | |
| 				return false; 
 | |
| 			}	
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		// Scenario I - use request owned POST data - toForm already executed
 | |
| 		$model->setPostedData($_posted) // insert() or update() dbInsert();
 | |
| 			->save(true, $forceSave);
 | |
| 
 | |
| 	    if ($id)
 | |
| 	    {
 | |
| 	        $new_data = $model->getData();
 | |
| 
 | |
| 	        if($changes = array_diff_assoc($new_data, $old_data))
 | |
| 	        {
 | |
| 	            $old_changed_data = array_intersect_key($old_data, $changes);
 | |
| 				$this->backupToHistory($this->table, $this->getPrimaryName(), $id, 'update', $old_changed_data);
 | |
| 	        }
 | |
| 
 | |
| 	    }
 | |
| 
 | |
| 
 | |
| 
 | |
| 	//	if(!empty($_POST))
 | |
| 		{
 | |
| 
 | |
| 		}
 | |
| 			
 | |
| 		// Scenario II - inner model sanitize
 | |
| 		//$this->getModel()->setPosted($this->convertToData($_POST, null, false, true);
 | |
| 
 | |
| 		// Take action based on use choice after success
 | |
| 		if(!$this->getModel()->hasError())
 | |
| 		{
 | |
| 			// callback (if any)
 | |
| 			$new_data 		= $model->getData();
 | |
| 			$id 			= $model->getId();
 | |
| 
 | |
| 			e107::getAddonConfig('e_admin',null,'process', $this, $id);
 | |
| 
 | |
| 			// Trigger Admin-ui event. 'post'
 | |
| 			if($triggerName = $this->getEventTriggerName( $this->getEventName(), $_posted['etrigger_submit'],'after')) // 'created' or 'updated';
 | |
| 			{
 | |
| 				$this->triggerEvent($triggerName, $_posted, $old_data, $id);
 | |
| 			}
 | |
| 
 | |
| 			if($callbackAfter && method_exists($this, $callbackAfter))
 | |
| 			{
 | |
| 				$this->$callbackAfter($new_data, $old_data, $id);
 | |
| 			}
 | |
| 			$model->setMessages(true); //FIX - move messages (and session messages) to the default stack
 | |
| 			$this->doAfterSubmit($model->getId(), $noredirectAction);
 | |
| 			return true;
 | |
| 		}
 | |
| 		elseif($callbackError && method_exists($this, $callbackError))
 | |
| 		{
 | |
| 			// suppress messages if callback returns TRUE
 | |
| 			if($this->$callbackError($_posted, $old_data, $model->getId()) !== true)
 | |
| 			{
 | |
| 				// Copy model messages to the default message stack
 | |
| 				$model->setMessages();
 | |
| 			}
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		// Copy model messages to the default message stack
 | |
| 		$model->setMessages();
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Trigger 2 events. The $triggerName event, and a matching generic admin_ui_xxxxx event.
 | |
| 	 * @param string $triggerName
 | |
| 	 * @param array $_posted
 | |
| 	 * @param array $old_data
 | |
| 	 * @param int $id
 | |
| 	 * @return false|mixed
 | |
| 	 */
 | |
| 	protected function triggerEvent($triggerName, $_posted, $old_data, $id)
 | |
| 	{
 | |
| 		unset($_posted['etrigger_submit'], $_posted['__after_submit_action'], $_posted['submit_value'], $_posted['e-token']);
 | |
| 
 | |
| 		$pid = $this->getPrimaryName();
 | |
| 		$_posted[$pid] = $id;    // add in the primary ID field.
 | |
| 
 | |
| 		$table = $this->getTableName();
 | |
| 		$pname = $this->getPluginName();
 | |
| 
 | |
| 		if($pname === 'core') // Handler 'core' plugin value.
 | |
| 		{
 | |
| 			$convert = array(
 | |
| 				'news_category' => 'news',
 | |
| 				'news'          => 'news',
 | |
| 				'page'          => 'page',
 | |
| 				'page_chapters' => 'page',
 | |
| 				'user'          => 'user'
 | |
| 			);
 | |
| 
 | |
| 			if(!empty($convert[$table]))
 | |
| 			{
 | |
| 				$pname = $convert[$table];
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$eventData = array( // use $_posted as it may include unsaved data.
 | |
| 			'newData'   => $_posted,
 | |
| 			'oldData'   => $old_data,
 | |
| 			'id'        => $id,
 | |
| 			'table'     => $table,
 | |
| 			'plugin'    => $pname,
 | |
| 		);
 | |
| 
 | |
| 		$this->_log('Triggering Event: ' . $triggerName);
 | |
| 
 | |
| 		$tmp = explode('_', $triggerName);
 | |
| 		$name = end($tmp);
 | |
| 
 | |
| 		$adminTriggerName = 'admin_ui_'.$name;
 | |
| 
 | |
| 
 | |
| 		e107::getMessage()->addDebug('Event triggers fired (<b>' .$triggerName. '</b>, <b>' . $adminTriggerName."</b>) 
 | |
| 		<a class='e-expandit' href='#view-event-data-".$name."'>Toggle data</a>
 | |
| 		<div id='view-event-data-".$name."' class='e-hideme'>" . 	print_a($eventData, true). '</div>'
 | |
| 		);
 | |
| 
 | |
| 		if($halt = e107::getEvent()->trigger($adminTriggerName, $eventData))
 | |
| 		{
 | |
| 			return $halt;
 | |
| 		}
 | |
| 
 | |
| 		return e107::getEvent()->trigger($triggerName, $eventData);
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 *  Return a custom event trigger name
 | |
| 	 * @param null $type  Usually 'Create' or 'Update'
 | |
| 	 * @param string $when ' before or after
 | |
| 	 * @return bool|string
 | |
| 	 */
 | |
| 	public function getEventTriggerName($name, $type=null, $when='before')
 | |
| 	{
 | |
| 		if(empty($name) || empty($type))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		if($when === 'after')
 | |
| 		{
 | |
| 			$type .= 'd'; // ie. 'created' or 'updated'.
 | |
| 		}
 | |
| 		
 | |
| 		return 'admin_'.strtolower($name).'_'.strtolower($type);
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Modifies a list query with search parameters and filters.
 | |
| 	 *
 | |
| 	 * @param string $listQry       The base list query string to be modified.
 | |
| 	 * @param string $searchTerm    The search term to filter records.
 | |
| 	 * @param string $filterOptions The filter options for customizing the query.
 | |
| 	 * @param string $tablePath     The table path for the query.
 | |
| 	 * @param string $tableFrom     The primary table being queried.
 | |
| 	 * @param string $primaryName   The primary key name of the table.
 | |
| 	 * @param mixed  $raw           Raw data that can influence query behavior.
 | |
| 	 * @param mixed  $orderField    The field by which to order results.
 | |
| 	 * @param mixed  $qryAsc        Sort order for the query (ascending or descending).
 | |
| 	 * @param mixed  $forceFrom     Forced starting position in the query.
 | |
| 	 * @param int    $qryFrom       Starting offset for the query.
 | |
| 	 * @param mixed  $forceTo       Forced ending position in the query.
 | |
| 	 * @param int    $perPage       The number of results per page.
 | |
| 	 * @param mixed  $qryField      Specific query field(s) to filter.
 | |
| 	 * @param mixed  $isfilter      Determines if a specific filter is applied.
 | |
| 	 * @param mixed  $handleAction  Custom action handler for the search process.
 | |
| 	 * @return string|false|array
 | |
| 	 */
 | |
| 	public function _modifyListQrySearch(string|null $listQry, string $searchTerm, string $filterOptions, string $tablePath,  string $tableFrom, string|null $primaryName, $raw, $orderField, $qryAsc, $forceFrom, int $qryFrom, $forceTo, int $perPage, $qryField,  $isfilter, $handleAction)
 | |
| 	{
 | |
| 		$generateTest = false;
 | |
| 		$tp       = e107::getParser();
 | |
| 		$fields   = $this->getFields();
 | |
| 		$joinData = $this->getJoinData();
 | |
| 
 | |
| 		$this->listQry = $listQry;
 | |
| 
 | |
| 		$tableSFieldsArr = array(); // FROM for main table
 | |
| 		$tableSJoinArr   = array(); // FROM for join tables
 | |
| 		$filter          = array();
 | |
| 		$searchQry       = array();
 | |
| 		$filterFrom      = array();
 | |
| 
 | |
| 		$searchTerm   = $tp->toDB($searchTerm);
 | |
| 		$searchQuery  = $this->fixSearchWildcards($searchTerm);
 | |
| 		$searchFilter = $this->_parseFilterRequest($filterOptions);
 | |
| 
 | |
| 		$listQry = $this->listQry; // check for modification during parseFilterRequest();
 | |
| 
 | |
| 		$debugData = [
 | |
| 			'uri'    => e_REQUEST_URI,
 | |
| 		    'methodInvocation' => [
 | |
| 		        'listQry'      => (string) $listQry,
 | |
| 		        'searchTerm'   => $searchTerm,
 | |
| 		        'filterOptions'=> $filterOptions,
 | |
| 		        'tablePath'    => $tablePath,
 | |
| 		        'tableFrom'    => $tableFrom,
 | |
| 		        'primaryName'  => $primaryName,
 | |
| 		        'raw'          => (bool) $raw,
 | |
| 		        'orderField'   => $orderField,
 | |
| 		        'qryAsc'       => $qryAsc,
 | |
| 		        'forceFrom'    => $forceFrom,
 | |
| 		        'qryFrom'      => $qryFrom,
 | |
| 		        'forceTo'      => $forceTo,
 | |
| 		        'perPage'      => $perPage,
 | |
| 		        'qryField'     => $qryField,
 | |
| 		        'isfilter'     => $isfilter,
 | |
| 		        'handleAction' => $handleAction
 | |
| 		    ],
 | |
| 		    'preProcessedData' => [
 | |
| 		        'fields'   => $fields,
 | |
| 		        'joinData' => $joinData,
 | |
| 		        'listOrder' => $this->listOrder,
 | |
| 		    ],
 | |
| 		    'intermediateStates' => [
 | |
| 		        'searchTerm'  => $searchTerm,
 | |
| 		        'searchQuery' => $searchQuery,
 | |
| 		        'searchFilter'=> $searchFilter,
 | |
| 		        'listQry'     => $this->listQry
 | |
| 		    ]
 | |
| 		];
 | |
| 
 | |
| 
 | |
| 
 | |
| 		if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 		{
 | |
| 			e107::getMessage()->addDebug('searchQuery: <b>' . $searchQuery . '</b>');
 | |
| 		}
 | |
| 
 | |
| 		if($searchFilter && is_array($searchFilter))
 | |
| 		{
 | |
| 
 | |
| 			list($filterField, $filterValue) = $searchFilter;
 | |
| 
 | |
| 			if($filterField && $filterValue !== '' && isset($fields[$filterField]))
 | |
| 			{
 | |
| 				if(!empty($fields[$filterField]['data']))
 | |
| 				{
 | |
| 					$_dataType  = $fields[$filterField]['data'];
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					$_dataType  = 'str';
 | |
| 					e107::getMessage()->addInfo('Field <b>' . $filterField . '</b> has no data type. Using default <b>str</b>.');
 | |
| 				}
 | |
| 
 | |
| 				$_fieldType = $fields[$filterField]['type'];
 | |
| 
 | |
| 				if($_fieldType === 'comma' || $_fieldType === 'checkboxes' || $_fieldType === 'userclasses' || ($_fieldType === 'dropdown' && !empty($fields[$filterField]['writeParms']['multiple'])))
 | |
| 				{
 | |
| 					$_dataType = 'set';
 | |
| 				}
 | |
| 
 | |
| 				switch($_dataType)
 | |
| 				{
 | |
| 					case 'set':
 | |
| 						$searchQry[] = "FIND_IN_SET('" . $tp->toDB($filterValue) . "', " . $fields[$filterField]['__tableField'] . ')';
 | |
| 					break;
 | |
| 
 | |
| 					case 'int':
 | |
| 					case 'integer':
 | |
| 						if($_fieldType === 'datestamp') // Past Month, Past Year etc.
 | |
| 						{
 | |
| 							$tmp            = explode('__', $filterOptions);
 | |
| 							$dateSearchType = $tmp[2];
 | |
| 
 | |
| 							if($filterValue > time())
 | |
| 							{
 | |
| 								if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 								{
 | |
| 									e107::getMessage()->addDebug("[$dateSearchType] Between now and " . date(DATE_RFC822, $filterValue));
 | |
| 								}
 | |
| 
 | |
| 								$searchQry[] = $fields[$filterField]['__tableField'] . ' > ' . time();
 | |
| 								$searchQry[] = $fields[$filterField]['__tableField'] . ' < ' . (int) $filterValue;
 | |
| 							}
 | |
| 							else // THIS X, FUTURE
 | |
| 							{
 | |
| 
 | |
| 								$endOpts = [
 | |
| 									'today'     => strtotime('+24 hours', $filterValue),
 | |
| 									'thisweek'  => strtotime('+1 week', $filterValue),
 | |
| 									'thismonth' => strtotime('+1 month', $filterValue),
 | |
| 									'thisyear'  => strtotime('+1 year', $filterValue),
 | |
| 								];
 | |
| 
 | |
| 								$end = $endOpts[$dateSearchType] ?? time();
 | |
| 
 | |
| 								if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 								{
 | |
| 									e107::getMessage()->addDebug("[$dateSearchType] Between " . date(DATE_RFC822, $filterValue) . " and " . date(DATE_RFC822, $end));
 | |
| 								}
 | |
| 
 | |
| 								$searchQry[] = $fields[$filterField]['__tableField'] . ' > ' . (int) $filterValue;
 | |
| 								$searchQry[] = $fields[$filterField]['__tableField'] . ' < ' . $end;
 | |
| 							}
 | |
| 
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							$searchQry[] = $fields[$filterField]['__tableField'] . ' = ' . (int) $filterValue;
 | |
| 						}
 | |
| 					break;
 | |
| 
 | |
| 
 | |
| 					default: // string usually.
 | |
| 
 | |
| 						if($filterValue === '_ISEMPTY_')
 | |
| 						{
 | |
| 							$searchQry[] = $fields[$filterField]['__tableField'] . " = '' ";
 | |
| 						}
 | |
| 
 | |
| 						else
 | |
| 						{
 | |
| 
 | |
| 							if($_fieldType === 'method') // More flexible filtering.
 | |
| 							{
 | |
| 
 | |
| 								$searchQry[] = $fields[$filterField]['__tableField'] . ' LIKE "%' . $tp->toDB($filterValue) . '%"';
 | |
| 							}
 | |
| 							else
 | |
| 							{
 | |
| 
 | |
| 								$searchQry[] = $fields[$filterField]['__tableField'] . " = '" . $tp->toDB($filterValue) . "'";
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 						//exit;
 | |
| 					break;
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 			//echo 'type= '. $fields[$filterField]['data'];
 | |
| 			//	print_a($fields[$filterField]);
 | |
| 		}
 | |
| 		elseif($searchFilter && is_string($searchFilter))
 | |
| 		{
 | |
| 
 | |
| 			// filter callbacks could add to WHERE clause
 | |
| 			$searchQry[] = $searchFilter;
 | |
| 		}
 | |
| 
 | |
| 		if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 		{
 | |
| 			e107::getMessage()->addDebug(print_a($searchQry, true));
 | |
| 		}
 | |
| 
 | |
| 		$className = get_class($this);
 | |
| 
 | |
| 		// main table should select everything
 | |
| 		$tableSFieldsArr[] = $tablePath . '*';
 | |
| 		foreach($fields as $key => $var)
 | |
| 		{
 | |
| 			// disabled or system
 | |
| 			if((!empty($var['nolist']) && empty($var['filter'])) || empty($var['type']) || empty($var['data']))
 | |
| 			{
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			// select FROM... for main table
 | |
| 			if(!empty($var['alias']) && !empty($var['__tableField']))
 | |
| 			{
 | |
| 				$tableSFieldsArr[] = $var['__tableField'];
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 			if($this->_isSearchField($var, $searchQuery))
 | |
| 			{
 | |
| 				// Search for customer filter handler.
 | |
| 				$customSearchMethod = 'handle' . $handleAction . eHelper::camelize($key) . 'Search';
 | |
| 
 | |
| 				$args                = array($searchTerm);
 | |
| 
 | |
| 				e107::getMessage()->addDebug('Searching for custom search method: ' . $className . '::' . $customSearchMethod . '(' . implode(', ', $args) . ')');
 | |
| 
 | |
| 				if (method_exists($this, $customSearchMethod)) // callback handling
 | |
| 				{
 | |
| 					e107::getMessage()->addDebug('Executing custom search callback <strong>' . $className . '::' . $customSearchMethod . '(' . implode(', ', $args) . ')</strong>');
 | |
| 
 | |
| 					$filter[] = call_user_func_array([$this, $customSearchMethod], $args);
 | |
| 					continue;
 | |
| 				}
 | |
| 				elseif (property_exists($this, $customSearchMethod)) // check for dynamic property
 | |
| 				{
 | |
| 
 | |
| 					e107::getMessage()->addDebug('Using custom search property <strong>' . $className . '::$' . $customSearchMethod . '</strong>');
 | |
| 
 | |
| 					$filter[] = call_user_func_array($this->$customSearchMethod, $args);
 | |
| 
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 
 | |
| 				if($var['data'] === 'int' || $var['data'] === 'integer' || $var['type'] === 'int' || $var['type'] === 'integer')
 | |
| 				{
 | |
| 					if(is_numeric($searchQuery))
 | |
| 					{
 | |
| 						$filter[] = $var['__tableField'] . ' = ' . $searchQuery;
 | |
| 					}
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				if($var['type'] === 'ip')
 | |
| 				{
 | |
| 					$ipSearch = e107::getIPHandler()->ipEncode($searchQuery);
 | |
| 					if(!empty($ipSearch))
 | |
| 					{
 | |
| 						$filter[] = $var['__tableField'] . " LIKE '%" . $ipSearch . "%'";
 | |
| 					}
 | |
| 					// Continue below for BC check also.
 | |
| 				}
 | |
| 
 | |
| 
 | |
| 				if(strpos($searchQuery, '|') === false  ) // search multiple words across fields.
 | |
| 				{
 | |
| 					$filter[] = $var['__tableField'] . " LIKE '%" . $searchQuery . "%'";
 | |
| 				}
 | |
| 
 | |
| 
 | |
| 				if($isfilter)
 | |
| 				{
 | |
| 					$filterFrom[] = $var['__tableField'];
 | |
| 
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if(strpos($searchQuery, '|') !== false) // search multiple words across fields.
 | |
| 		{
 | |
| 			$tmp = explode('|', $searchQuery);
 | |
| 
 | |
| 			if(count($tmp) < 4) // avoid excessively long query.
 | |
| 			{
 | |
| 
 | |
| 					foreach($tmp as $splitSearchQuery)
 | |
| 					{
 | |
| 						if(!empty($splitSearchQuery))
 | |
| 						{
 | |
| 							$multiWordSearch = [];
 | |
| 							foreach($fields as $key => $var)
 | |
| 							{
 | |
| 								if(!$this->_isSearchField($var, $splitSearchQuery) || $var['data'] === 'int' || $var['data'] === 'integer' || $var['type'] === 'int' || $var['type'] === 'integer')
 | |
| 								{
 | |
| 									continue;
 | |
| 								}
 | |
| 
 | |
| 								$multiWordSearch[] = $var['__tableField'] . " LIKE '%" . $splitSearchQuery . "%'";
 | |
| 							}
 | |
| 							$searchQry[] = '('.implode(' OR ', $multiWordSearch).')';
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 	//	fwrite(STDOUT, __LINE__ . print_r($filter,true) . "\n");
 | |
| 
 | |
| 
 | |
| 		if(strpos($filterOptions, 'searchfield__') === 0) // search in specific field, so remove the above filters.
 | |
| 		{
 | |
| 			$filter = array(); // reset filter.
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		//	if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 		{
 | |
| 			//	e107::getDebug()->log(print_a($filter,true));
 | |
| 			// e107::getMessage()->addInfo(print_a($filter,true));
 | |
| 		}
 | |
| 
 | |
| 		if($isfilter)
 | |
| 		{
 | |
| 			if(!$filterFrom)
 | |
| 			{
 | |
| 				return false;
 | |
| 			}
 | |
| 			$tableSFields = implode(', ', $filterFrom);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$tableSFields = $tableSFieldsArr ? implode(', ', $tableSFieldsArr) : $tablePath . '*';
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		$jwhere = array();
 | |
| 		$joins  = array();
 | |
| 		//file_put_contents(e_LOG.'uiAjaxResponseSFields.log', $tableSFields."\n\n", FILE_APPEND);
 | |
| 		//file_put_contents(e_LOG.'uiAjaxResponseFields.log', print_r($this->getFields(), true)."\n\n", FILE_APPEND);
 | |
| 		if($joinData)
 | |
| 		{
 | |
| 			$qry = 'SELECT ' . $tableSFields;
 | |
| 			foreach($joinData as $jtable => $tparams)
 | |
| 			{
 | |
| 				// Select fields
 | |
| 				if(!$isfilter)
 | |
| 				{
 | |
| 					$fields = vartrue($tparams['fields']);
 | |
| 					if($fields === '*')
 | |
| 					{
 | |
| 						$tableSJoinArr[] = "{$tparams['__tablePath']}*";
 | |
| 					}
 | |
| 					elseif($fields)
 | |
| 					{
 | |
| 						$tableSJoinArr[] = $fields;
 | |
| 						/*$fields = explode(',', $fields);
 | |
| 						foreach ($fields as $field)
 | |
| 						{
 | |
| 							$qry .= ", {$tparams['__tablePath']}`".trim($field).'`';
 | |
| 						}*/
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				// Prepare Joins
 | |
| 				$joins[] = '
 | |
| 					' . vartrue($tparams['joinType'], 'LEFT JOIN') . " {$tparams['__tableFrom']} ON " . (vartrue($tparams['leftTable']) ? $tparams['leftTable'] . '.' : $tablePath) . '`' . vartrue($tparams['leftField']) . "` = {$tparams['__tablePath']}`" . vartrue($tparams['rightField']) . '`' . (vartrue($tparams['whereJoin']) ? ' ' . $tparams['whereJoin'] : '');
 | |
| 
 | |
| 				// Prepare Where
 | |
| 				if(!empty($tparams['where']))
 | |
| 				{
 | |
| 					$jwhere[] = $tparams['where'];
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 			//From
 | |
| 			$qry .= $tableSJoinArr ? ', ' . implode(', ', $tableSJoinArr) . ' FROM ' . $tableFrom : ' FROM ' . $tableFrom;
 | |
| 
 | |
| 			// Joins
 | |
| 			if(count($joins) > 0)
 | |
| 			{
 | |
| 				$qry .= "\n" . implode("\n", $joins);
 | |
| 			}
 | |
| 		}
 | |
| 		else    // default listQry
 | |
| 		{
 | |
| 			if(!empty($listQry))
 | |
| 			{
 | |
| 				$qry = $this->parseCustomListQry($listQry);
 | |
| 			}
 | |
| 			elseif($this->sortField && $this->sortParent && !deftrue('e_DEBUG_TREESORT')) // automated 'tree' sorting.
 | |
| 			{
 | |
| 				//	$qry = "SELECT SQL_CALC_FOUND_ROWS a. *, CASE WHEN a.".$this->sortParent." = 0 THEN a.".$this->sortField." ELSE b.".$this->sortField." + (( a.".$this->sortField.")/1000) END AS treesort FROM `#".$this->table."` AS a LEFT JOIN `#".$this->table."` AS b ON a.".$this->sortParent." = b.".$this->pid;
 | |
| 				$qry = $this->getParentChildQry(true);
 | |
| 				//$this->listOrder	= '_treesort '; // .$this->sortField;
 | |
| 				//	$this->orderStep    = ($this->orderStep === 1) ? 100 : $this->orderStep;
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$qry = 'SELECT ' . $tableSFields . ' FROM ' . $tableFrom;
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		// group field - currently auto-added only if there are joins
 | |
| 		$groupField = '';
 | |
| 		if($joins && $primaryName)
 | |
| 		{
 | |
| 			$groupField = $tablePath . $primaryName;
 | |
| 		}
 | |
| 
 | |
| 		// appended to GROUP BY when true.
 | |
| 		if(!empty($this->listGroup))
 | |
| 		{
 | |
| 			$groupField = $this->listGroup;
 | |
| 		}
 | |
| 
 | |
| 		if($raw)
 | |
| 		{
 | |
| 			$rawData = array(
 | |
| 				'joinWhere'     => $jwhere,
 | |
| 				'filter'        => $filter,
 | |
| 				'listQrySql'    => $this->listQrySql,
 | |
| 				'filterFrom'    => $filterFrom,
 | |
| 				'search'        => $searchQry,
 | |
| 				'tableFromName' => $tableFrom,
 | |
| 			);
 | |
| 
 | |
| 
 | |
| 			$rawData['tableFrom']  = $tableSFieldsArr;
 | |
| 			$rawData['joinsFrom']  = $tableSJoinArr;
 | |
| 			$rawData['joins']      = $joins;
 | |
| 			$rawData['groupField'] = $groupField;
 | |
| 			$rawData['orderField'] = isset($fields[$orderField]) ? $fields[$orderField]['__tableField'] : '';
 | |
| 			$rawData['orderType']  = $qryAsc === 'desc' ? 'DESC' : 'ASC';
 | |
| 			$rawData['limitFrom']  = $forceFrom === false ? $qryFrom : (int) $forceFrom;
 | |
| 			$rawData['limitTo']    = $forceTo === false ? $perPage : (int) $forceTo;
 | |
| 
 | |
| 			return $rawData;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		// join where
 | |
| 		if(count($jwhere) > 0)
 | |
| 		{
 | |
| 			$searchQry[] = ' (' . implode(' AND ', $jwhere) . ' )';
 | |
| 		}
 | |
| 		// filter where
 | |
| 		if(count($filter) > 0)
 | |
| 		{
 | |
| 			$searchQry[] = ' ( ' . implode(' OR ', $filter) . ' ) ';
 | |
| 		}
 | |
| 
 | |
| 		// more user added sql
 | |
| 		if(isset($this->listQrySql['db_where']) && $this->listQrySql['db_where'])
 | |
| 		{
 | |
| 			if(is_array($this->listQrySql['db_where']))
 | |
| 			{
 | |
| 				$searchQry[] = implode(' AND ', $this->listQrySql['db_where']);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$searchQry[] = $this->listQrySql['db_where'];
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		// where query
 | |
| 		if(count($searchQry) > 0)
 | |
| 		{
 | |
| 			// add more where details on the fly via $this->listQrySql['db_where'];
 | |
| 			$qry .= (strripos($qry, 'where') === false) ? ' WHERE ' : ' AND '; // Allow 'where' in custom listqry
 | |
| 			$qry .= implode(' AND ', $searchQry);
 | |
| 
 | |
| 			// Disable tree (use flat list instead) when filters are applied
 | |
| 			// Implemented out of necessity under https://github.com/e107inc/e107/issues/3204
 | |
| 			// Horrible hack, but only needs this one line of additional code
 | |
| 			if($treemodel = $this->getTreeModel())
 | |
| 			{
 | |
| 				$treemodel->setParam('sort_parent', null);
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		// GROUP BY if needed
 | |
| 		if($groupField)
 | |
| 		{
 | |
| 			$qry .= ' GROUP BY ' . $groupField;
 | |
| 		}
 | |
| 
 | |
| 		// only when no custom order is required
 | |
| 		if($this->listOrder && !$qryField && !$qryAsc)
 | |
| 		{
 | |
| 			$qry .= ' ORDER BY ' . $this->listOrder;
 | |
| 		}
 | |
| 		elseif($this->listOrder !== false)
 | |
| 		{
 | |
| 			$orderField = !empty($qryField) ? $qryField : $this->getDefaultOrderField();
 | |
| 			$orderDef   = ($qryAsc === null ? $this->getDefaultOrder() : $qryAsc);
 | |
| 			if(isset($fields[$orderField]) && strpos($this->listQry, 'ORDER BY') == false) //override ORDER using listQry (admin->sitelinks)
 | |
| 			{
 | |
| 				// no need of sanitize - it's found in field array
 | |
| 				$qry .= ' ORDER BY ' . $fields[$orderField]['__tableField'] . ' ' . (strtolower($orderDef) === 'desc' ? 'DESC' : 'ASC');
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if(isset($this->filterQry)) // custom query on filter. (see downloads plugin)
 | |
| 		{
 | |
| 			$qry = $this->filterQry;
 | |
| 		}
 | |
| 
 | |
| 		if($forceTo !== false || $perPage)
 | |
| 		{
 | |
| 			$from = $forceFrom === false ? $qryFrom : (int) $forceFrom;
 | |
| 			if($forceTo === false)
 | |
| 			{
 | |
| 				$forceTo = $perPage;
 | |
| 			}
 | |
| 			$qry .= ' LIMIT ' . $from . ', ' . (int) $forceTo;
 | |
| 		}
 | |
| 
 | |
| 		// Debug Filter Query.
 | |
| 		if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 		{
 | |
| 			e107::getMessage()->addDebug('QRY=' . str_replace('#', MPREFIX, $qry));
 | |
| 		}
 | |
| 		//	 echo $qry.'<br />';
 | |
| 		// print_a($this->fields);
 | |
| 
 | |
| 		$this->_log('listQry: ' . str_replace('#', MPREFIX, $qry));
 | |
| 
 | |
| 		// JSON encode the debug data
 | |
| 		$debugData['intermediateStates']['listQryBeforeFinal'] = $listQry;
 | |
| 
 | |
| 		$debugData['expected'] = $qry;
 | |
| 		$jsonDebugInfo = json_encode($debugData, JSON_PRETTY_PRINT);
 | |
| 
 | |
| 		// Optionally log the JSON data to a file for inspection
 | |
| 		if($generateTest && !e107::isCli())
 | |
| 		{
 | |
| 			$path = e_BASE."e107_tests/tests/_data/e_admin_ui/_modifyListQrySearch/".sha1($jsonDebugInfo).".json";
 | |
| 			if(file_put_contents($path, $jsonDebugInfo . PHP_EOL, FILE_APPEND))
 | |
| 			{
 | |
| 				e107::getMessage()->addDebug('Saved test info to ' . $path);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		// Print to the debug interface (optional, can overload logs)
 | |
| 			if(E107_DEBUG_LEVEL == E107_DBG_SQLQUERIES)
 | |
| 			{
 | |
| 				e107::getMessage()->addDebug('<pre>' . $jsonDebugInfo . '</pre>');
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 		return $qry;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Checks whether the field array should be searched oor not.
 | |
| 	 * @param array $var
 | |
| 	 * @param string $searchQuery
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	private function _isSearchField($field, $searchQuery): bool
 | |
| 	{
 | |
| 		$searchable_types = array('text', 'textarea', 'bbarea', 'url', 'ip', 'tags', 'email', 'int', 'integer', 'str', 'safestr', 'string', 'number'); //method? 'user',
 | |
| 
 | |
| 		if($field['type'] === 'method' && !empty($field['data']) && ($field['data'] === 'string' || $field['data'] === 'str' || $field['data'] === 'safestr' || $field['data'] === 'int'))
 | |
| 		{
 | |
| 			$searchable_types[] = 'method';
 | |
| 		}
 | |
| 
 | |
| 		return !empty($field['__tableField']) && trim($searchQuery) !== '' && in_array($field['type'], $searchable_types);
 | |
| 
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  */
 | |
| class e_admin_ui extends e_admin_controller_ui
 | |
| {
 | |
| 
 | |
| 	protected $fieldTypes = array();
 | |
| 	protected $dataFields = array();
 | |
| 	protected $fieldInputTypes = array();
 | |
| 	protected $validationRules = array();
 | |
| 
 | |
| 	protected $table;
 | |
| 	protected $pid;
 | |
| 	protected $listQry;
 | |
| 	protected $editQry;
 | |
| 	protected $sortField;
 | |
| 	protected $sortParent;
 | |
| 	protected $orderStep;
 | |
| 	protected $treePrefix;
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Markup to be auto-inserted before List filter
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	public $preFilterMarkup = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Markup to be auto-inserted after List filter
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	public $postFilterMarkup = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Markup to be auto-inserted at the top of Create form
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	public $headerCreateMarkup = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Markup to be auto-inserted at the bottom of Create form
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	public $footerCreateMarkup = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Markup to be auto-inserted at the top of Update form
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	public $headerUpdateMarkup = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Markup to be auto-inserted at the bottom of Update form
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	public $footerUpdateMarkup = '';
 | |
| 
 | |
| 	/**
 | |
| 	 * Show confirm screen before (batch/single) delete
 | |
| 	 * @var boolean
 | |
| 	 */
 | |
| 	public $deleteConfirmScreen = false;
 | |
| 	
 | |
| 	/**
 | |
| 	 * Confirm screen custom message
 | |
| 	 * @var string
 | |
| 	 */
 | |
| 	public $deleteConfirmMessage;
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor
 | |
| 	 * @param e_admin_request $request
 | |
| 	 * @param e_admin_response $response
 | |
| 	 * @param array $params [optional]
 | |
| 	 */
 | |
| 	public function __construct($request, $response, $params = array())
 | |
| 	{
 | |
| 		$this->setDefaultAction($request->getDefaultAction());
 | |
| 		$params['enable_triggers'] = true; // override
 | |
| 
 | |
| 		parent::__construct($request, $response, $params);
 | |
| 
 | |
| 		if(!$this->pluginName)
 | |
| 		{
 | |
| 			$this->pluginName = 'core';
 | |
| 		}
 | |
| 
 | |
| 	/*	$ufieldpref = $this->getUserPref();
 | |
| 		if($ufieldpref)
 | |
| 		{
 | |
| 			$this->fieldpref = $ufieldpref;
 | |
| 		}*/
 | |
| 
 | |
| 		$this->addTitle($this->pluginTitle)->parseAliases();
 | |
| 
 | |
| 		$this->initAdminAddons();
 | |
| 
 | |
| 
 | |
| 		if($help = $this->renderHelp())
 | |
| 		{
 | |
| 			if(!empty($help))
 | |
| 			{
 | |
| 				e107::setRegistry('core/e107/adminui/help',$help);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	private function initAdminAddons()
 | |
| 	{
 | |
| 		$tmp = e107::getAddonConfig('e_admin', null, 'config', $this);
 | |
| 
 | |
| 		if(empty($tmp))
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		$opts = null;
 | |
| 
 | |
| 		foreach($tmp as $plug=>$config)
 | |
| 		{
 | |
| 
 | |
| 			$form = e107::getAddon($plug, 'e_admin', $plug. '_admin_form'); // class | false.
 | |
| 
 | |
| 			if(!empty($config['fields']))
 | |
| 			{
 | |
| 				if(!empty($this->fields['options']))
 | |
| 				{
 | |
| 					$opts = $this->fields['options'];
 | |
| 					unset($this->fields['options']);
 | |
| 				}
 | |
| 
 | |
| 				foreach($config['fields'] as $k=>$v)
 | |
| 				{
 | |
| 					$v['data'] = false; // disable data-saving to db table. .
 | |
| 
 | |
| 					$fieldName = 'x_'.$plug.'_'.$k;
 | |
| 					e107::getDebug()->log($fieldName. ' initiated by ' .$plug);
 | |
| 
 | |
| 					if($v['type'] === 'method' && method_exists($form,$fieldName))
 | |
| 					{
 | |
| 						$v['method'] = $plug. '_admin_form::' .$fieldName;
 | |
| 						//echo "Found method ".$fieldName." in ".$plug."_menu_form";
 | |
| 						//echo $form->$fieldName();
 | |
| 					}
 | |
| 
 | |
| 
 | |
| 					$this->fields[$fieldName] = $v; // ie. x_plugin_key
 | |
| 
 | |
| 				}
 | |
| 
 | |
| 				if(!empty($opts)) // move options field to the end.
 | |
| 				{
 | |
| 					$this->fields['options'] = $opts;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if(!empty($config['batchOptions']))
 | |
| 			{
 | |
| 				$opts = array();
 | |
| 				foreach($config['batchOptions'] as $k=>$v)
 | |
| 				{
 | |
| 					$fieldName = 'batch_'.$plug.'_'.$k;
 | |
| 
 | |
| 					$opts[$fieldName] = $v; // ie. x_plugin_key
 | |
| 
 | |
| 				}
 | |
| 
 | |
| 				$batchCat = deftrue('LAN_PLUGIN_'.strtoupper($plug).'_NAME', $plug);
 | |
| 				$this->batchOptions[$batchCat] = $opts;
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 			if(!empty($config['tabs']))
 | |
| 			{
 | |
| 				foreach($config['tabs'] as $t=>$tb)
 | |
| 				{
 | |
| 					$this->tabs[$t] = $tb;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Catch fieldpref submit
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function ListEcolumnsTrigger()
 | |
| 	{
 | |
| 		$this->setTriggersEnabled(false); //disable further triggering
 | |
| 		parent::manageColumns();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Detect if a batch function has been fired.
 | |
| 	 * @param $batchKey
 | |
| 	 * @return bool
 | |
| 	 */
 | |
| 	public function batchTriggered($batchKey)
 | |
| 	{
 | |
| 		return (!empty($_POST['e__execute_batch']) && (varset($_POST['etrigger_batch']) == $batchKey));
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Catch batch submit
 | |
| 	 * @param string $batch_trigger
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	public function ListBatchTrigger($batch_trigger)
 | |
| 	{
 | |
| 		$this->setPosted('etrigger_batch');
 | |
| 
 | |
| 		if($this->getPosted('etrigger_cancel'))
 | |
| 		{
 | |
| 			$this->setPosted(array());
 | |
| 			return; // always break on cancel!
 | |
| 		}
 | |
| 		$this->deleteConfirmScreen = true; // Confirm screen ALWAYS enabled when multi-deleting!
 | |
| 
 | |
| 		// proceed ONLY if there is no other trigger, except delete confirmation
 | |
| 		if($batch_trigger && !$this->hasTrigger(array('etrigger_delete_confirm')))
 | |
| 		{
 | |
| 			$this->_handleListBatch($batch_trigger);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 		/**
 | |
| 	 * Catch batch submit
 | |
| 	 * @param string $batch_trigger
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	public function GridBatchTrigger($batch_trigger)
 | |
| 	{
 | |
| 		$this->setPosted('etrigger_batch');
 | |
| 
 | |
| 		if($this->getPosted('etrigger_cancel'))
 | |
| 		{
 | |
| 			$this->setPosted(array());
 | |
| 			return; // always break on cancel!
 | |
| 		}
 | |
| 		$this->deleteConfirmScreen = true; // Confirm screen ALWAYS enabled when multi-deleting!
 | |
| 
 | |
| 		// proceed ONLY if there is no other trigger, except delete confirmation
 | |
| 		if($batch_trigger && !$this->hasTrigger(array('etrigger_delete_confirm')))
 | |
| 		{
 | |
| 			$this->_handleListBatch($batch_trigger);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Batch delete trigger
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function handleListDeleteBatch($selected)
 | |
| 	{
 | |
| 		
 | |
| 		$tp = e107::getParser();
 | |
| 		
 | |
| 		if(!$this->getBatchDelete())
 | |
| 		{
 | |
| 			e107::getMessage()->add(LAN_UI_BATCHDEL_ERROR, E_MESSAGE_WARNING);
 | |
| 			return;
 | |
| 		}
 | |
| 		if($this->deleteConfirmScreen)
 | |
| 		{
 | |
| 			if(!$this->getPosted('etrigger_delete_confirm'))
 | |
| 			{
 | |
| 				// ListPage will show up confirmation screen
 | |
| 				$this->setPosted('delete_confirm_value', implode(',', $selected));
 | |
| 				return;
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				// already confirmed, resurrect selected values
 | |
| 				$selected = explode(',', $this->getPosted('delete_confirm_value'));
 | |
| 				foreach ($selected as $i => $_sel) 
 | |
| 				{
 | |
| 					$selected[$i] = preg_replace('/[^\w\-:.]/', '', $_sel);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// delete one by one - more control, less performance
 | |
| 		// pass  afterDelete() callback to tree delete method
 | |
| 		$set_messages = true;
 | |
| 		$delcount = 0;
 | |
| 		$nfcount = 0;
 | |
| 		foreach ($selected as $id)
 | |
| 		{
 | |
| 			$data = array();
 | |
| 			$model = $this->getTreeModel()->getNode($id);
 | |
| 			if($model)
 | |
| 			{
 | |
| 				$data = $model->getData();
 | |
| 
 | |
| 				if($this->table !== 'admin_history')
 | |
| 				{
 | |
| 					$this->backupToHistory($this->table, $this->pid, $id, 'delete', $data);
 | |
| 				}
 | |
| 
 | |
| 				if($this->beforeDelete($data, $id))
 | |
| 				{
 | |
| 					$check = $this->getTreeModel()->delete($id);
 | |
| 					if($check)
 | |
| 					{
 | |
| 						$delcount++;
 | |
| 					}
 | |
| 					if(!$this->afterDelete($data, $id, $check))
 | |
| 					{
 | |
| 						$set_messages = false;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$set_messages = true;	
 | |
| 				$nfcount++; 
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		//$this->getTreeModel()->delete($selected);
 | |
| 		if($set_messages) 
 | |
| 		{
 | |
| 			$this->getTreeModel()->setMessages();
 | |
| 			// FIXME lan
 | |
| 			if($delcount)
 | |
| 			{
 | |
| 				e107::getMessage()->addSuccess($tp->lanVars(LAN_UI_DELETED, $delcount, true));
 | |
| 			}
 | |
| 			if($nfcount)
 | |
| 			{
 | |
| 				e107::getMessage()->addError($tp->lanVars(LAN_UI_DELETED_FAILED, $nfcount, true));
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		//$this->redirect();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Batch copy trigger
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function handleListCopyBatch($selected)
 | |
| 	{
 | |
| 		// Batch Copy
 | |
| 
 | |
| 		$res = $this->getTreeModel()->copy($selected);
 | |
| 		// callback
 | |
| 		$this->afterCopy($res, $selected);
 | |
| 		// move messages to default stack 
 | |
| 		$this->getTreeModel()->setMessages();
 | |
| 		// send messages to session
 | |
| 		e107::getMessage()->moveToSession();
 | |
| 		// redirect
 | |
| 		$this->redirect();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Batch Export trigger
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function handleListExportBatch($selected)
 | |
| 	{
 | |
| 		// Batch Copy
 | |
| 		$res = $this->getTreeModel()->export($selected);
 | |
| 		// callback
 | |
| 	//	$this->afterCopy($res, $selected);
 | |
| 		// move messages to default stack
 | |
| 		$this->getTreeModel()->setMessages();
 | |
| 		// send messages to session
 | |
| 		e107::getMessage()->moveToSession();
 | |
| 		// redirect
 | |
| 		$this->redirect();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 	 * Batch Export trigger
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function handleListSefgenBatch($selected, $field, $value)
 | |
| 	{
 | |
| 
 | |
| 		$tree = $this->getTreeModel();
 | |
| 		$c= 0;
 | |
| 		foreach($selected as $id)
 | |
|         {
 | |
|         	if(!$tree->hasNode($id))
 | |
|         	{
 | |
|         		e107::getMessage()->addError('Item #ID '.htmlspecialchars($id).' not found.');
 | |
|         		continue;
 | |
|         	}
 | |
| 
 | |
|         	$model = $tree->getNode($id);
 | |
| 
 | |
|         	$name = $model->get($value);
 | |
| 
 | |
|         	$sef = eHelper::title2sef($name,'dashl');
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
|         	$model->set($field, $sef);
 | |
| 
 | |
| 
 | |
|         	$model->save();
 | |
| 
 | |
|         	$data = $model->getData();
 | |
| 
 | |
|         	if($model->isModified())
 | |
| 	        {
 | |
| 	           	$this->getModel()->setData($data)->save(false,true);
 | |
| 		        $c++;
 | |
| 	        }
 | |
|         }
 | |
| 
 | |
| 
 | |
| 
 | |
| 		$caption = e107::getParser()->lanVars(LAN_UI_BATCH_BOOL_SUCCESS, $c, true);
 | |
| 		e107::getMessage()->addSuccess($caption);
 | |
| 
 | |
| 	//	e107::getMessage()->moveToSession();
 | |
| 		// redirect
 | |
| 	//	$this->redirect();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
|     /** 
 | |
|      * Batch URL trigger
 | |
|      * @param array $selected
 | |
|      * @return void
 | |
|      */
 | |
|     protected function handleListUrlBatch($selected)
 | |
|     {
 | |
|         if($this->_add2nav($selected))
 | |
| 		{
 | |
| 			e107::getMessage()->moveToSession();
 | |
| 			$this->redirect();
 | |
| 		}
 | |
|     }
 | |
| 
 | |
| 
 | |
| 	/** TODO
 | |
| 	 * Batch Featurebox Transfer
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function handleListFeatureboxBatch($selected)
 | |
| 	{
 | |
| 		 if($this->_add2featurebox($selected))
 | |
| 		{
 | |
| 			e107::getMessage()->moveToSession();
 | |
| 			$this->redirect();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $selected
 | |
| 	 * @return false|int
 | |
| 	 */
 | |
| 	protected function _add2nav($selected)
 | |
| 	{
 | |
| 		if(empty($selected))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}// TODO warning message
 | |
| 		
 | |
| 		if(!is_array($selected))
 | |
| 		{
 | |
| 			$selected = array($selected);
 | |
| 		}
 | |
| 
 | |
|         $sql        = e107::getDb();
 | |
| 		$urlData	= $this->getUrl();
 | |
| 		$allData 	= $this->getTreeModel()->url($selected, array('sc' => true), true);
 | |
| 
 | |
|         e107::getMessage()->addDebug('Using Url Route:'.$urlData['route']);   
 | |
|         
 | |
| 		$scount = 0;
 | |
|         foreach($allData as $id => $data)
 | |
|         {
 | |
|             $name = $data['name'];
 | |
|             $desc = $data['description'];
 | |
|             
 | |
|             $link = $data['url'];
 | |
|             
 | |
|             $link = str_replace('{e_BASE}', '', $link); // TODO temporary here, discuss
 | |
|             
 | |
|             // _FIELD_TYPES auto created inside mysql handler now
 | |
|             $linkArray = array(
 | |
|                 'link_name'         => $name, 
 | |
|                 'link_url'          => $link,
 | |
|                 'link_description'  => e107::getParser()->toDB($desc), // retrieved field type is string, we might need todb here
 | |
|                 'link_button'       => '',
 | |
|                 'link_category'     => 255, // Using an unassigned template rather than inactive link-class, since other inactive links may already exist. 
 | |
|                 'link_order'        => 0,
 | |
|                 'link_parent'       => 0,
 | |
|                 'link_open'         => '',
 | |
|                 'link_class'        => 0,
 | |
|                 'link_sefurl'		=> e107::getParser()->toDB($urlData['route'].'?'.$id),
 | |
|             );
 | |
|             
 | |
|             $res = $sql->insert('links', $linkArray);
 | |
|             
 | |
|             if($res !== FALSE)
 | |
|             {
 | |
| 				e107::getMessage()->addSuccess(LAN_CREATED. ': ' .LAN_NAVIGATION. ': ' .($name ? $name : 'n/a'));
 | |
| 				$scount++; 
 | |
|             }
 | |
|             else 
 | |
|             {
 | |
|                 if($sql->getLastErrorNumber())
 | |
|                 {
 | |
| 					e107::getMessage()->addError(LAN_CREATED_FAILED. ': ' .LAN_NAVIGATION. ': ' .$name. ': ' .LAN_SQL_ERROR);
 | |
|                     e107::getMessage()->addDebug('SQL Link Creation Error #'.$sql->getLastErrorNumber().': '.$sql->getLastErrorText());
 | |
|                 }
 | |
| 				else
 | |
| 				{
 | |
| 					e107::getMessage()->addError(LAN_CREATED_FAILED. ': ' .LAN_NAVIGATION. ': ' .$name. ': ' .LAN_UNKNOWN_ERROR);//Unknown Error
 | |
| 				}
 | |
|             }
 | |
| 
 | |
|         }
 | |
|         
 | |
| 		if($scount > 0)
 | |
| 		{
 | |
| 			e107::getMessage()->addSuccess(LAN_CREATED. ' (' .$scount. ') ' .LAN_NAVIGATION_LINKS);
 | |
| 			e107::getMessage()->addSuccess("<a class='btn btn-small btn-primary' href='".e_ADMIN_ABS."links.php?searchquery=&filter_options=link_category__255'>".LAN_CONFIGURE. ' ' .LAN_NAVIGATION. '</a>');
 | |
| 			return $scount;        
 | |
| 		}
 | |
|         
 | |
|         return false; 
 | |
|  
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $selected
 | |
| 	 * @return false|int
 | |
| 	 */
 | |
| 	protected function _add2featurebox($selected)
 | |
| 	{
 | |
| 		// FIX - don't allow if plugin not installed
 | |
| 		if(!e107::isInstalled('featurebox'))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 		
 | |
| 		if(empty($selected))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}// TODO warning message
 | |
| 		
 | |
| 		if(!is_array($selected))
 | |
| 		{
 | |
| 			$selected = array($selected);
 | |
| 		}
 | |
| 
 | |
|         $sql        = e107::getDb();
 | |
| 		$tree = $this->getTreeModel();
 | |
| 		$urlData = $this->getTreeModel()->url($selected, array('sc' => true));
 | |
| 		$data = $this->featurebox;
 | |
| 		
 | |
| 		$scount = 0;
 | |
| 		$category = 0;
 | |
| 
 | |
|         foreach($selected as $id)
 | |
|         {
 | |
|         	if(!$tree->hasNode($id)) 
 | |
|         	{
 | |
|         		e107::getMessage()->addError('Item #ID '.htmlspecialchars($id).' not found.');
 | |
|         		continue; // TODO message
 | |
|         	} 
 | |
|         	
 | |
|         	$model = $tree->getNode($id);
 | |
|             if($data['url'] === true)
 | |
| 			{
 | |
| 				$url = $urlData[$id];
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$url = $model->get($data['url']);
 | |
| 			}
 | |
| 			$name = $model->get($data['name']);
 | |
| 			
 | |
| 			$category = e107::getDb()->retrieve('featurebox_category', 'fb_category_id', "fb_category_template='unassigned'");
 | |
| 			
 | |
|             $fbArray = array (
 | |
|                 	'fb_title' 		=> $name, 
 | |
|      				'fb_text' 		=> $model->get($data['description']), 
 | |
| 					'fb_image' 		=> vartrue($data['image']) ? $model->get($data['image']) : '',
 | |
| 					'fb_imageurl'	=> $url, 
 | |
| 					'fb_class' 		=> isset($data['visibility']) && $data['visibility'] !== false ? $model->get($data['visibility']) : e_UC_ADMIN,
 | |
| 					'fb_template' 	=> 'default',
 | |
| 					'fb_category' 	=> $category, // TODO popup - choose category
 | |
| 					'fb_order' 		=> $scount, 
 | |
|             );
 | |
| 
 | |
|             $res = $sql->insert('featurebox', $fbArray);
 | |
| 
 | |
|             if($res !== FALSE)
 | |
|             {
 | |
| 				e107::getMessage()->addSuccess(LAN_CREATED. ': ' .LAN_PLUGIN_FEATUREBOX_NAME. ': ' .($name ? $name : 'n/a'));
 | |
| 				$scount++; 
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 if($sql->getLastErrorNumber())
 | |
|                 {
 | |
| 					e107::getMessage()->addError(LAN_CREATED_FAILED. ': ' .LAN_PLUGIN_FEATUREBOX_NAME. ': ' .$name. ': ' .LAN_SQL_ERROR);
 | |
| 					e107::getMessage()->addDebug('SQL Featurebox Creation Error #'.$sql->getLastErrorNumber().': '.$sql->getLastErrorText());
 | |
|                 }  
 | |
| 				else
 | |
| 				{
 | |
| 					e107::getMessage()->addError(LAN_CREATED_FAILED. ': ' .$name. ': ' .LAN_UNKNOWN_ERROR);
 | |
| 				}
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         if($scount > 0)
 | |
|         {
 | |
| 			e107::getMessage()->addSuccess(LAN_CREATED. ' (' .$scount. ') ' .defset('LAN_PLUGIN_FEATUREBOX_NAME'));
 | |
| 			e107::getMessage()->addSuccess("<a class='btn btn-small btn-primary' href='".e_PLUGIN_ABS."featurebox/admin_config.php?searchquery=&filter_options=fb_category__{$category}' ".LAN_CONFIGURE. ' ' .LAN_PLUGIN_FEATUREBOX_NAME. '</a>');
 | |
| 			return $scount;        
 | |
|         }
 | |
|         
 | |
|         return false; 
 | |
|  
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 	
 | |
| 	/**
 | |
| 	 * Batch boolean trigger
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function handleListBoolBatch($selected, $field, $value)
 | |
| 	{
 | |
| 		$cnt = $this->getTreeModel()->batchUpdate($field, $value, $selected, $value, false);
 | |
| 		if($cnt)
 | |
| 		{
 | |
| 			$caption = e107::getParser()->lanVars(LAN_UI_BATCH_BOOL_SUCCESS, $cnt, true);
 | |
| 			$this->getTreeModel()->addMessageSuccess($caption);
 | |
| 		}
 | |
| 		$this->getTreeModel()->setMessages();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Batch boolean reverse trigger
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	protected function handleListBoolreverseBatch($selected, $field)
 | |
| 	{
 | |
| 		$tree = $this->getTreeModel();
 | |
| 		$cnt = $tree->batchUpdate($field, "1-{$field}", $selected, null, false);
 | |
| 		if($cnt)
 | |
| 		{
 | |
| 			$caption = e107::getParser()->lanVars(LAN_UI_BATCH_REVERSED_SUCCESS, $cnt, true);
 | |
| 			$tree->addMessageSuccess($caption);
 | |
| 			//sync models
 | |
| 			$tree->loadBatch(true);
 | |
| 		}
 | |
| 		$this->getTreeModel()->setMessages();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $selected
 | |
| 	 * @param $field
 | |
| 	 * @param $value
 | |
| 	 * @param $type
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function handleCommaBatch($selected, $field, $value, $type)
 | |
| 	{
 | |
| 		$tree = $this->getTreeModel();
 | |
| 		$rcnt = 0;
 | |
| 		$cnt = $rcnt;
 | |
| 		$value = e107::getParser()->toDB($value);
 | |
| 		
 | |
| 		switch ($type) 
 | |
| 		{
 | |
| 			case 'attach':
 | |
| 			case 'deattach':
 | |
| 				$this->_setModel();
 | |
| 				foreach ($selected as $key => $id) 
 | |
| 				{
 | |
| 					$node = $tree->getNode($id);
 | |
| 					if(!$node)
 | |
| 					{
 | |
| 						continue;
 | |
| 					}
 | |
| 					$val = $node->get($field);
 | |
| 					
 | |
| 					if(empty($val))
 | |
| 					{
 | |
| 						$val = array();
 | |
| 					}
 | |
| 					elseif(!is_array($val))
 | |
| 					{
 | |
| 						$val = explode(',', $val);
 | |
| 					}
 | |
| 					
 | |
| 					if($type === 'deattach')
 | |
| 					{
 | |
| 						$search = array_search($value, $val);
 | |
| 						if($search === false)
 | |
| 						{
 | |
| 							continue;
 | |
| 						}
 | |
| 						unset($val[$search]);
 | |
| 						sort($val);
 | |
| 						$val = implode(',', $val);
 | |
| 						$node->set($field, $val);
 | |
| 						$check = $this->getModel()->setData($node->getData())->save(false, true);
 | |
| 						
 | |
| 						if($check === false)
 | |
| 						{
 | |
| 							$this->getModel()->setMessages();
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							$rcnt++;
 | |
| 						}
 | |
| 						continue;
 | |
| 					}
 | |
| 					
 | |
| 					// attach it
 | |
| 					if(in_array($value, $val) === false)
 | |
| 					{
 | |
| 						$val[] = $value; 
 | |
| 						sort($val);
 | |
| 						$val = implode(',', array_unique($val));
 | |
| 						$node->set($field, $val);
 | |
| 						$check = $this->getModel()->setData($node->getData())->save(false, true);
 | |
| 						if($check === false)
 | |
| 						{
 | |
| 							$this->getModel()->setMessages();
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							$cnt++;
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				$this->_model = null;
 | |
| 			break;
 | |
| 				
 | |
| 			case 'addAll':
 | |
| 				if(!empty($value))
 | |
| 				{
 | |
| 					if(is_array($value))
 | |
| 					{ 
 | |
| 						sort($value);	
 | |
| 						$value = implode(',', array_map('trim', $value));
 | |
| 					}
 | |
| 					
 | |
| 					$cnt = $this->getTreeModel()->batchUpdate($field, $value, $selected, true);
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					$this->getTreeModel()->addMessageWarning(LAN_UPDATED_FAILED)->setMessages();//"Comma list is empty, aborting."
 | |
| 					$this->getTreeModel()->addMessageDebug(LAN_UPDATED_FAILED. ': Comma list is empty, aborting.')->setMessages();
 | |
| 				}
 | |
| 			break;
 | |
| 				
 | |
| 			case 'clearAll':
 | |
| 				$allowed = !is_array($value) ? explode(',', $value) : $value;
 | |
| 				if(!$allowed)
 | |
| 				{
 | |
| 					$rcnt = $this->getTreeModel()->batchUpdate($field, '', $selected, '');
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					$this->_setModel();
 | |
| 					foreach ($selected as $key => $id) 
 | |
| 					{
 | |
| 						$node = $tree->getNode($id);
 | |
| 						if(!$node)
 | |
| 						{
 | |
| 							continue;
 | |
| 						}
 | |
| 						
 | |
| 						$val = $node->get($field);
 | |
| 						
 | |
| 						// nothing to do
 | |
| 						if(empty($val))
 | |
| 						{
 | |
| 							break;
 | |
| 						}
 | |
| 						elseif(!is_array($val))
 | |
| 						{
 | |
| 							$val = explode(',', $val);
 | |
| 						}
 | |
| 						
 | |
| 						// remove only allowed, see userclass
 | |
| 						foreach ($val as $_k => $_v) 
 | |
| 						{
 | |
| 							if(in_array($_v, $allowed))
 | |
| 							{
 | |
| 								unset($val[$_k]);
 | |
| 							}
 | |
| 						}
 | |
| 						
 | |
| 						sort($val);
 | |
| 						$val = !empty($val) ? implode(',', $val) : '';
 | |
| 						$node->set($field, $val);
 | |
| 						$check = $this->getModel()->setData($node->getData())->save(false, true);
 | |
| 						
 | |
| 						if($check === false)
 | |
| 						{
 | |
| 							$this->getModel()->setMessages();
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							$rcnt++;
 | |
| 						}
 | |
| 					}
 | |
| 					$this->_model = null;
 | |
| 				}
 | |
| 
 | |
| 				// format for proper message
 | |
| 				$value = implode(',', $allowed);
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		if($cnt)
 | |
| 		{
 | |
| 			$vttl = $this->getUI()->renderValue($field, $value, $this->getFieldAttr($field));
 | |
| 			$caption = e107::getParser()->lanVars(LAN_UI_BATCH_UPDATE_SUCCESS, array('x'=>$vttl, 'y'=>$cnt), true);
 | |
| 			$this->getTreeModel()->addMessageSuccess($caption);
 | |
| 		}
 | |
| 		elseif($rcnt)
 | |
| 		{
 | |
| 			$vttl = $this->getUI()->renderValue($field, $value, $this->getFieldAttr($field));
 | |
| 			$caption = e107::getParser()->lanVars(LAN_UI_BATCH_DEATTACH_SUCCESS, array('x'=>$vttl, 'y'=>$cnt), true);
 | |
| 			$this->getTreeModel()->addMessageSuccess($caption);
 | |
| 		}
 | |
| 		$this->getTreeModel()->setMessages();
 | |
| 	}
 | |
| 
 | |
| 	protected function handleGridSearchfieldFilter($selected)
 | |
| 	{
 | |
| 		return $this->handleListSearchfieldFilter($selected);
 | |
| 	}
 | |
| 
 | |
| 	protected function handleGridDeleteBatch($selected)
 | |
| 	{
 | |
| 		return $this->handleListDeleteBatch($selected);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Method to generate "Search in Field" query.
 | |
| 	 * @param $selected
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	protected function handleListSearchfieldFilter($selected)
 | |
| 	{
 | |
| 		$string = $this->getQuery('searchquery');
 | |
| 
 | |
| 
 | |
| 
 | |
| 		if(empty($string))
 | |
| 		{
 | |
| 			return '';
 | |
| 		}
 | |
| 
 | |
| 		return $selected. " LIKE '%".e107::getParser()->toDB($string)."%' "; // array($selected, $this->getQuery('searchquery'));
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Batch default (field) trigger
 | |
| 	 * @param array $selected
 | |
| 	 * @return int|null
 | |
| 	 */
 | |
| 	protected function handleListBatch($selected, $field, $value)
 | |
| 	{
 | |
| 		// special exceptions
 | |
| 		
 | |
| 		if($value === '#delete') // see admin->users
 | |
| 		{
 | |
| 			$val = "''";
 | |
| 			$value = '(empty)';
 | |
| 		}	
 | |
| 		elseif($value === '#null')
 | |
| 		{
 | |
| 			$val = null;
 | |
| 			$value = '(empty)';
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$val = "'".$value."'";	
 | |
| 		}
 | |
| 		
 | |
| 		if($field === 'options') // reserved field type. see: admin -> media-manager - batch rotate image.
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 		$cnt = $this->getTreeModel()->batchUpdate($field, $val, $selected, true, false);
 | |
| 		if($cnt)
 | |
| 		{
 | |
| 			$vttl = $this->getUI()->renderValue($field, $value, $this->getFieldAttr($field));
 | |
| 			$msg = e107::getParser()->lanVars(LAN_UI_BATCH_UPDATE_SUCCESS, array('x' => $vttl, 'y' => $cnt), true);
 | |
| 			$this->getTreeModel()->addMessageSuccess($msg);
 | |
| 			// force reload the collection from DB, fix some issues as 'observer' is executed before the batch handler
 | |
| 			$this->getTreeModel()->setParam('db_query', $this->_modifyListQry(false, false, false, false, $this->listQry))->loadBatch(true);
 | |
| 		}
 | |
| 		$this->getTreeModel()->setMessages();
 | |
| 		return $cnt;
 | |
| 	}
 | |
| 
 | |
| 	public function GridDeleteTrigger($posted)
 | |
| 	{
 | |
| 		$this->ListDeleteTrigger($posted);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Catch delete submit
 | |
| 	 * @param $posted
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	public function ListDeleteTrigger($posted)
 | |
| 	{
 | |
| 		if($this->getPosted('etrigger_cancel'))
 | |
| 		{
 | |
| 			$this->setPosted(array());
 | |
| 			return; // always break on cancel!
 | |
| 		}
 | |
| 
 | |
| 		$id = (int) key($posted);
 | |
| 		if($this->deleteConfirmScreen && !$this->getPosted('etrigger_delete_confirm'))
 | |
| 		{
 | |
| 			// forward data to delete confirm screen
 | |
| 			$this->setPosted('delete_confirm_value', $id);
 | |
| 			return; // User confirmation expected
 | |
| 		}
 | |
| 
 | |
| 		$this->setTriggersEnabled(false);
 | |
| 		$data = array();
 | |
| 		$model = $this->getTreeModel()->getNode($id); //FIXME - this has issues with being on a page other than the 1st. 
 | |
| 		if($model)
 | |
| 		{
 | |
| 			$data = $model->getData();
 | |
| 
 | |
| 			if($this->table !== 'admin_history')
 | |
| 			{
 | |
| 				$this->backupToHistory($this->table, $this->pid, $id,'delete',$data);
 | |
| 			}
 | |
| 
 | |
| 			if($this->beforeDelete($data, $id))
 | |
| 			{
 | |
| 
 | |
| 				if($triggerName = $this->getEventTriggerName($this->getEventName(),'delete')) // trigger for before.
 | |
| 				{
 | |
| 
 | |
| 					if($halt = $this->triggerEvent($triggerName, null, $data, $id))
 | |
| 					{
 | |
| 						$this->getTreeModel()->setMessages();
 | |
| 						return null;
 | |
| 					} 	
 | |
| 				}
 | |
| 				
 | |
| 				$check = $this->getTreeModel()->delete($id);
 | |
| 				 		 
 | |
| 				if($this->afterDelete($data, $id, $check))
 | |
| 				{
 | |
| 					if($triggerName = $this->getEventTriggerName($this->getEventName(), 'deleted')) // trigger for after.
 | |
| 					{
 | |
| 						$this->triggerEvent($triggerName, null, $data, $id);
 | |
| 					}
 | |
| 					
 | |
| 					$this->getTreeModel()->setMessages();
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$this->getTreeModel()->setMessages();// errors
 | |
| 			}
 | |
| 		}
 | |
| 		else  //FIXME - this is a fall-back for the BUG which causes model to fail on all list pages other than the 1st
 | |
| 		{ 
 | |
| 			//echo "Couldn't get Node for ID: ".$id;
 | |
| 			// exit; 
 | |
| 			e107::getMessage()->addDebug('Model Failure Fallback in use!! ID: '.$id.' file: '.__FILE__. ' line: ' .__LINE__ ,'default',true);
 | |
| 			$check = $this->getTreeModel()->delete($id);
 | |
| 			return;			
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined pre-delete logic
 | |
| 	 */
 | |
| 	public function beforeDelete($data, $id)
 | |
| 	{
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined after-delete logic
 | |
| 	 */
 | |
| 	public function afterDelete($deleted_data, $id, $deleted_check)
 | |
| 	{
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * List action header
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function ListHeader()
 | |
| 	{
 | |
| 		//e107::js('core','core/tabs.js','prototype');
 | |
| 		//e107::js('core','core/admin.js','prototype');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * List action observer
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function ListObserver()
 | |
| 	{
 | |
| 		if($ufieldpref = $this->getUserPref())
 | |
| 		{
 | |
| 			$this->fieldpref = $ufieldpref;
 | |
| 		}
 | |
| 
 | |
| 		$table = $this->getTableName();
 | |
| 		if(empty($table))
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		$this->getTreeModel()->setParam('db_query', $this->_modifyListQry(false, false, false, false, $this->listQry))->loadBatch();
 | |
| 
 | |
| 		$this->addTitle();
 | |
| 
 | |
| 		
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Grid action observer
 | |
| 	 */
 | |
| 	public function GridObserver()
 | |
| 	{
 | |
| 
 | |
| 		$table = $this->getTableName();
 | |
| 		if(empty($table))
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 		$this->getTreeModel()->setParam('db_query', $this->_modifyListQry(false, false, false, false, $this->listQry))->loadBatch();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Filter response ajax page
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function FilterAjaxPage()
 | |
| 	{
 | |
| 		return $this->renderAjaxFilterResponse($this->listQry); //listQry will be used only if available
 | |
| 	}
 | |
| 	
 | |
| 	/**
 | |
| 	 * Inline edit action
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function InlineAjaxPage()
 | |
| 	{
 | |
| 		$this->logajax('Inline Ajax Triggered');
 | |
| 
 | |
| 		$protocol = (isset($_SERVER['SERVER_PROTOCOL']) ? $_SERVER['SERVER_PROTOCOL'] : 'HTTP/1.0');
 | |
| 		if(!vartrue($_POST['name']) || !vartrue($this->fields[$_POST['name']]))
 | |
| 		{
 | |
| 			header($protocol.': 404 Not Found', true, 404);
 | |
| 			header('Status: 404 Not Found', true, 404);
 | |
| 			echo LAN_FIELD. ': ' .$this->fields[$_POST['name']]. ': ' .LAN_NOT_FOUND; // Field: x: not found!
 | |
| 			$this->logajax('Field not found');
 | |
| 			return;
 | |
| 		}
 | |
| 		
 | |
| 		$_name = $_POST['name'];
 | |
| 		$_value = $_POST['value'];
 | |
| 		$_token = $_POST['token'];
 | |
| 
 | |
| 		$parms = $this->fields[$_name]['readParms'] ? $this->fields[$_name]['readParms'] : '';
 | |
| 		if(!is_array($parms))
 | |
| 		{
 | |
| 			parse_str($parms, $parms);
 | |
| 		}
 | |
| 		if(!empty($parms['editable']))
 | |
| 		{
 | |
| 			$this->fields[$_name]['inline'] = true;
 | |
| 		}
 | |
| 		
 | |
| 		if(!empty($this->fields[$_name]['noedit']) || !empty($this->fields[$_name]['nolist']) || empty($this->fields[$_name]['inline']) || empty($_token) || !password_verify(session_id(),$_token))
 | |
| 		{
 | |
| 			header($protocol.': 403 Forbidden', true, 403);
 | |
| 			header('Status: 403 Forbidden', true, 403);
 | |
| 			echo ADLAN_86; //Forbidden
 | |
| 
 | |
| 			$result = var_export($this->fields[$_name], true);
 | |
| 
 | |
| 			$problem = array();
 | |
| 			$problem['noedit'] = !empty($this->fields[$_name]['noedit']) ? 'yes' : 'no';
 | |
| 			$problem['nolist'] = !empty($this->fields[$_name]['nolist']) ? 'yes' : 'no';
 | |
| 			$problem['inline'] = empty($this->fields[$_name]['inline']) ? 'yes' : 'no';
 | |
| 			$problem['token'] = empty($_token) ? 'yes' : 'no';
 | |
| 			$problem['password'] = !password_verify(session_id(),$_token) ? 'yes' : 'no';
 | |
| 
 | |
| 			$result .= "\nForbidden Caused by: ".print_r($problem,true);
 | |
| 			$this->logajax("Forbidden\nAction:".$this->getAction()."\nField (".$_name."):\n".$result);
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 		
 | |
| 		$model = $this->getModel()->load($this->getId());
 | |
| 		$_POST = array(); //reset post
 | |
| 		$_POST[$_name] = $_value; // set current field only
 | |
| 		$_POST['etrigger_submit'] = 'update'; // needed for event trigger
 | |
| 
 | |
| 	//	print_r($_POST);
 | |
| 		
 | |
| 		// generic handler - same as regular edit form submit
 | |
| 
 | |
| 		$this->convertToData($_POST);
 | |
| 
 | |
| 		$model->setPostedData($_POST);
 | |
| 		$model->setParam('validateAvailable', true); // new param to control validate of available data only, reset on validate event
 | |
| 		// Do not update here! Because $old_data and $new_data will be the same in afterUpdate() methods.
 | |
| 		// Data will be saved in _manageSubmit() method.
 | |
| 		// $model->update(true);
 | |
| 
 | |
| 		if($model->hasError())
 | |
| 		{
 | |
| 			// using 400
 | |
| 			header($protocol.': 400 Bad Request', true, 400);
 | |
| 			header('Status: 400 Bad Request', true, 400);
 | |
| 			$this->logajax('Bad Request');
 | |
| 			// DEBUG e107::getMessage()->addError('Error test.', $model->getMessageStackName())->addError('Another error test.', $model->getMessageStackName());
 | |
| 
 | |
| 
 | |
| 			if(E107_DEBUG_LEVEL)
 | |
| 			{
 | |
| 				$message = e107::getMessage()->get('debug', $model->getMessageStackName(), true);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$message = e107::getMessage()->get('error', $model->getMessageStackName(), true);
 | |
| 			}
 | |
| 			
 | |
| 			if(!empty($message))
 | |
| 			{
 | |
| 				echo implode(' ', $message);
 | |
| 			}
 | |
| 			$this->logajax(implode(' ', $message));
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		//TODO ? afterInline trigger?
 | |
| 		$res = $this->_manageSubmit('beforeUpdate', 'afterUpdate', 'onUpdateError', 'edit');
 | |
| 	}
 | |
| 
 | |
| 	// Temporary - but useful. :-)
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $message
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function logajax($message)
 | |
| 	{
 | |
| 		if(e_DEBUG !== true)
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		$message = date('r')."\n".$message."\n";
 | |
| 		$message .= "\n_POST\n";
 | |
| 		$message .= print_r($_POST,true);
 | |
| 		$message .= "\n_GET\n";
 | |
| 		$message .= print_r($_GET,true);
 | |
| 
 | |
| 		$message .= '---------------';
 | |
| 		
 | |
| 		file_put_contents(e_LOG.'uiAjaxResponseInline.log', $message."\n\n", FILE_APPEND);
 | |
| 	}
 | |
| 	
 | |
| 	
 | |
| 	/**
 | |
| 	 * Drag-n-Drop sort action
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function SortAjaxPage()
 | |
| 	{
 | |
| 		if(!isset($_POST['all']) || empty($_POST['all']))
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 		if(!$this->sortField)
 | |
| 		{
 | |
| 			echo 'Missing sort field value';
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		$this->_log('Executing SortAjaxPage()');
 | |
| 		$sql    = e107::getDb();
 | |
| 
 | |
| 		if(!empty($this->sortParent)) // Force 100 positions for child when sorting with parent/child.
 | |
| 		{
 | |
| 			$this->orderStep = 100;
 | |
| 		}
 | |
| 		else // Reset all the order fields first.
 | |
| 		{
 | |
| 			$resetQry = $this->sortField ."= 999 WHERE 1"; // .$this->sortField;
 | |
| 			$sql->update($this->table, $resetQry );
 | |
| 			$this->_log('Sort Qry ('.$this->table.'): '.$resetQry);
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		$step   = $this->orderStep ? (int) $this->orderStep : 1;
 | |
| 		$from   = !empty($_GET['from']) ? (int) $_GET['from'] * $step : $step;
 | |
| 
 | |
| 		$c = $from;
 | |
| 		$updated = array();
 | |
| 
 | |
| 		foreach($_POST['all'] as $row)
 | |
| 		{
 | |
| 
 | |
| 			list($tmp,$id) = explode('-', $row, 2);
 | |
| 			$id = preg_replace('/[^\w\-:.]/', '', $id);
 | |
| 			if(!is_numeric($id))
 | |
| 			{
 | |
| 				$id = "'{$id}'";
 | |
| 			}
 | |
| 			$updateQry = $this->sortField." = {$c} WHERE ".$this->pid. ' = ' .$id;
 | |
| 
 | |
| 			if($sql->update($this->table, $updateQry) !==false)
 | |
| 			{
 | |
| 				$updated[] = '#' .$id. '  --  ' .$this->sortField. ' = ' .$c;
 | |
| 			}
 | |
| 
 | |
| 			$this->_log('Sort Qry ('.$this->table.'): '.$updateQry);
 | |
| 
 | |
| 			$c += $step;
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		if(!empty($this->sortParent) && !empty($this->sortField) )
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		// Increment every record after the current page of records.
 | |
| 
 | |
| 		$changed = $c - $step;
 | |
| 		$qry = 'UPDATE `#' .$this->table. '` e, (SELECT @n := ' .($changed). ') m  SET e.' .$this->sortField. ' = @n := @n + ' .$step. ' WHERE ' .$this->sortField. ' > ' .($changed);
 | |
| 
 | |
| 		$result = $sql->gen($qry);
 | |
| 		$this->_log('Sort Qry: '.$qry);
 | |
| 
 | |
| 
 | |
| 		// ------------ Fix Child Order when parent is used. ----------------
 | |
| /*
 | |
| 		if(!empty($this->sortParent) && !empty($this->sortField) ) // Make sure there is space for at least 99
 | |
| 		{
 | |
| 			$parent = array();
 | |
| 
 | |
| 			$data2 = $sql->retrieve($this->table,$this->pid.','.$this->sortField,$this->sortParent .' = 0',true);
 | |
| 			foreach($data2 as $val)
 | |
| 			{
 | |
| 				$id = $val[$this->pid];
 | |
| 				$parent[$id] = $val[$this->sortField];
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 			$previous = 0;
 | |
| 
 | |
| 			$data = $sql->retrieve($this->table,'*',$this->sortParent.' != 0 ORDER BY '.$this->sortField,true);
 | |
| 
 | |
| 			foreach($data as $row)
 | |
| 			{
 | |
| 				$p = $row[$this->sortParent];
 | |
| 
 | |
| 				if($p != $previous)
 | |
| 				{
 | |
| 					$c = $parent[$p];
 | |
| 				}
 | |
| 
 | |
| 				$c++;
 | |
| 				$previous = $p;
 | |
| 
 | |
| 				//	echo "<br />".$row['forum_name']." with parent: ".$p." old: ".$row['forum_order']."  new: ".$c;
 | |
| 				$sql->update($this->table, $this->sortField . ' = '.$c.' WHERE '.$this->pid.' = '.intval($row[$this->pid]).' LIMIT 1');
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 		}
 | |
| */
 | |
| 		$this->afterSort($result, $_POST);
 | |
| 
 | |
| 	//	e107::getLog()->addDebug(print_r($_POST,true))->toFile('SortAjax','Admin-UI Ajax Sort Log', true);
 | |
| 	//	 e107::getLog()->addDebug(print_r($updated,true))->toFile('SortAjax','Admin-UI Ajax Sort Log', true);
 | |
| 	//	e107::getLog()->addDebug($qry)->toFile('SortAjax','Admin-UI Ajax Sort Log', true);
 | |
| 
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic List action page
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function ListPage()
 | |
| 	{
 | |
| 		if($this->deleteConfirmScreen && !$this->getPosted('etrigger_delete_confirm') && $this->getPosted('delete_confirm_value'))
 | |
| 		{
 | |
| 			// 'edelete_confirm_data' set by single/batch delete trigger
 | |
| 			return $this->getUI()->getConfirmDelete($this->getPosted('delete_confirm_value')); // User confirmation expected
 | |
| 		}
 | |
| 		return $this->getUI()->getList();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic List action page
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function GridPage()
 | |
| 	{
 | |
| 		if($this->deleteConfirmScreen && !$this->getPosted('etrigger_delete_confirm') && $this->getPosted('delete_confirm_value'))
 | |
| 		{
 | |
| 			// 'edelete_confirm_data' set by single/batch delete trigger
 | |
| 			return $this->getUI()->getConfirmDelete($this->getPosted('delete_confirm_value')); // User confirmation expected
 | |
| 		}
 | |
| 
 | |
| 		return $this->getUI()->getList(null,'grid');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * List action observer
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function ListAjaxObserver()
 | |
| 	{
 | |
| 	    if($ufieldpref = $this->getUserPref())
 | |
| 		{
 | |
| 			$this->fieldpref = $ufieldpref;
 | |
| 		}
 | |
| 
 | |
| 		$this->getTreeModel()->setParam('db_query', $this->_modifyListQry(false, false, 0, false, $this->listQry))->loadBatch();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * List action observer
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function GridAjaxObserver()
 | |
| 	{
 | |
| 		$this->ListAjaxObserver();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic List action page (Ajax)
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function ListAjaxPage()
 | |
| 	{
 | |
| 		return $this->getUI()->getList(true);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function GridAjaxPage()
 | |
| 	{
 | |
| 		return $this->getUI()->getList(true,'grid');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Edit observer
 | |
| 	 */
 | |
| 	public function EditObserver()
 | |
| 	{
 | |
| 		$this->getModel()->load($this->getId());
 | |
| 		$this->addTitle();
 | |
| 		$this->addTitle('#'.$this->getId()); // Inform user of which record is being edited.
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Create submit trigger
 | |
| 	 */
 | |
| 	public function EditCancelTrigger()
 | |
| 	{
 | |
| 		$this->redirectAction('list', 'id');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Edit submit trigger
 | |
| 	 */
 | |
| 	public function EditSubmitTrigger()
 | |
| 	{
 | |
| 		$this->_manageSubmit('beforeUpdate', 'afterUpdate', 'onUpdateError', 'edit');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Edit - send JS to page Header
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function EditHeader()
 | |
| 	{
 | |
| 		// e107::getJs()->requireCoreLib('core/admin.js');
 | |
| 		e107::js('core','core/admin.js','prototype');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Edit page
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function EditPage()
 | |
| 	{
 | |
| 		return $this->CreatePage();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Create observer
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function CreateObserver()
 | |
| 	{
 | |
| 		$this->setTriggersEnabled(true);
 | |
| 		$this->addTitle();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Create submit trigger
 | |
| 	 */
 | |
| 	public function CreateCancelTrigger()
 | |
| 	{
 | |
| 		$this->redirectAction('list', 'id');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Create submit trigger
 | |
| 	 */
 | |
| 	public function CreateSubmitTrigger()
 | |
| 	{
 | |
| 		$this->_manageSubmit('beforeCreate', 'afterCreate', 'onCreateError');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined pre-create logic, return false to prevent DB query execution
 | |
| 	 * @param $new_data
 | |
| 	 * @param $old_data
 | |
| 	 */
 | |
| 	public function beforeCreate($new_data, $old_data)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined after-create logic
 | |
| 	 * @param $new_data
 | |
| 	 * @param $old_data
 | |
| 	 * @param $id
 | |
| 	 */
 | |
| 	public function afterCreate($new_data, $old_data, $id)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined error handling, return true to suppress model messages
 | |
| 	 * @param $new_data
 | |
| 	 * @param $old_data
 | |
| 	 */
 | |
| 	public function onCreateError($new_data, $old_data)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined pre-update logic, return false to prevent DB query execution
 | |
| 	 * @param $new_data
 | |
| 	 * @param $old_data
 | |
| 	 * @param $id
 | |
| 	 */
 | |
| 	public function beforeUpdate($new_data, $old_data, $id)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined after-update logic
 | |
| 	 * @param $new_data
 | |
| 	 * @param $old_data
 | |
| 	 * @param $id
 | |
| 	 */
 | |
| 	public function afterUpdate($new_data, $old_data, $id)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined before pref saving logic
 | |
| 	 * @param $new_data
 | |
| 	 * @param $old_data
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	public function beforePrefsSave($new_data, $old_data)
 | |
| 	{
 | |
| 		return null;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined before pref saving logic
 | |
| 	 */
 | |
| 	public function afterPrefsSave()
 | |
| 	{
 | |
| 		return null;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
|     * User defined error handling, return true to suppress model messages
 | |
|     */
 | |
| 	public function onUpdateError($new_data, $old_data, $id)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined after-update logic
 | |
| 	 * @param mixed $result
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function afterCopy($result, $selected)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined after-sort logic
 | |
| 	 * @param mixed $result
 | |
| 	 * @param array $selected
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function afterSort($result, $selected)
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function renderHelp()
 | |
| 	{
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Create - send JS to page Header
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function CreateHeader()
 | |
| 	{
 | |
| 		// TODO - invoke it on className (not all textarea elements)
 | |
| 		//e107::getJs()->requireCoreLib('core/admin.js');
 | |
| 		e107::js('core','core/admin.js','prototype');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 *
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function CreatePage()
 | |
| 	{
 | |
| 		return $this->getUI()->getCreate();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function PrefsSaveTrigger()
 | |
| 	{
 | |
| 		$data = $this->getPosted();
 | |
| 
 | |
| 		$beforePref = $data;
 | |
| 		unset($beforePref['e-token'],$beforePref['etrigger_save']);
 | |
| 
 | |
| 		$tmp = $this->beforePrefsSave($beforePref, $this->getConfig()->getPref());
 | |
| 
 | |
| 		if(!empty($tmp))
 | |
| 		{
 | |
| 			$data = $tmp;
 | |
| 		}
 | |
| 
 | |
| 		foreach($this->prefs as $k=>$v) // fix for empty checkboxes - need to save a value.
 | |
| 		{
 | |
| 			if(!isset($data[$k]) && $v['data'] !== false && ($v['type'] === 'checkboxes' || $v['type'] === 'checkbox'))
 | |
| 			{
 | |
| 				$data[$k] = null;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		foreach($data as $key=>$val)
 | |
| 		{
 | |
| 
 | |
| 			if(!empty($this->prefs[$key]['multilan']))
 | |
| 			{
 | |
| 
 | |
| 				if(is_string($this->getConfig()->get($key))) // most likely upgraded to multilan=>true, so reset to an array structure.
 | |
| 				{
 | |
| 					$this->getConfig()->setPostedData($key, array(e_LANGUAGE => $val));
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					$lang = key($val);
 | |
| 					$value = $val[$lang];
 | |
| 					$this->getConfig()->setData($key.'/'.$lang, str_replace("'", ''', $value));
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$this->getConfig()->setPostedData($key, $val);
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		$this->getConfig()->save();
 | |
| 
 | |
| 		$this->afterPrefsSave();
 | |
| 
 | |
| /*
 | |
| 		$this->getConfig()
 | |
| 			->setPostedData($this->getPosted(), null, false)
 | |
| 			//->setPosted('not_existing_pref_test', 1)
 | |
| 			->save(true);
 | |
| */
 | |
| 
 | |
| 		$this->getConfig()->setMessages();
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function PrefsObserver()
 | |
| 	{
 | |
| 		$this->addTitle();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function PrefsPage()
 | |
| 	{
 | |
| 		return $this->getUI()->getSettings();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Parent overload
 | |
| 	 * @return e_admin_ui
 | |
| 	 */
 | |
| 	protected function parseAliases()
 | |
| 	{
 | |
| 		// parse table
 | |
| 		$tableName = $this->getTableName();
 | |
| 		if(strpos($tableName, '.') !== false)
 | |
| 		{
 | |
| 			$tmp = explode('.', $tableName, 2);
 | |
| 			$this->table = $tmp[1];
 | |
| 			$this->tableAlias = $tmp[0];
 | |
| 			unset($tmp);
 | |
| 		}
 | |
| 
 | |
| 		parent::parseAliases();
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return false
 | |
| 	 */
 | |
| 	public function getPrimaryName()
 | |
| 	{
 | |
| 		// Option for working with tables having no PID
 | |
| 		if($this->pid !== false && empty($this->pid) && !empty($this->fields))
 | |
| 		{
 | |
| 			$message = e107::getParser()->toHTML(LAN_UI_NOPID_ERROR,true);
 | |
| 			e107::getMessage()->add($message, E_MESSAGE_WARNING);
 | |
| 		}
 | |
| 
 | |
| 		return $this->pid;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $alias
 | |
| 	 * @param $prefix
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getTableName($alias = false, $prefix = false)
 | |
| 	{
 | |
| 		if($alias)
 | |
| 		{
 | |
| 			return ($this->tableAlias ? $this->tableAlias : '');
 | |
| 		}
 | |
| 		return ($prefix ? '#' : '').$this->table;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Validation rules retrieved from controller object
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getValidationRules()
 | |
| 	{
 | |
| 		return $this->validationRules;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Data Field array retrieved from controller object
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getDataFields()
 | |
| 	{
 | |
| 		return $this->dataFields;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Set read and write parms with drop-down-list array data (ie. type='dropdown')
 | |
| 	 * @param string $field
 | |
| 	 * @param array $array [optional]
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	public function setDropDown($field,$array) //TODO Have Miro check this.
 | |
| 	{
 | |
| 		$this->fields[$field]['readParms'] = $array;
 | |
| 		$this->fields[$field]['writeParms'] = $array;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Set Config object
 | |
| 	 * @return e_admin_ui
 | |
| 	 */
 | |
| 	protected function _setConfig()
 | |
| 	{
 | |
| 		$this->_pref = $this->pluginName === 'core' ? e107::getConfig() : e107::getPlugConfig($this->pluginName);
 | |
| 
 | |
| 		if($this->pluginName !== 'core' && !e107::isInstalled($this->pluginName))
 | |
| 		{
 | |
| 			$obj = get_class($this);
 | |
| 			e107::getMessage()->addWarning($obj. 'The plugin is not installed or $pluginName: is not valid. (' .$this->pluginName. ')'); // debug only.
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		$validateRules = array();
 | |
| 		$dataFields = $validateRules;
 | |
| 		foreach ($this->prefs as $key => $att)
 | |
| 		{
 | |
| 			// create dataFields array
 | |
| 			$dataFields[$key] = vartrue($att['data'], 'string');
 | |
| 
 | |
| 			// create validation array
 | |
| 			if(!empty($att['validate']))
 | |
| 			{
 | |
| 				$validateRules[$key] = array(($att['validate'] === true ? 'required' : $att['validate']), varset($att['rule']), $att['title'], varset($att['error'], $att['help']));
 | |
| 			}
 | |
| 			/* Not implemented in e_model yet
 | |
| 			elseif(vartrue($att['check']))
 | |
| 			{
 | |
| 				$validateRules[$key] = array($att['check'], varset($att['rule']), $att['title'], varset($att['error'], $att['help']));
 | |
| 			}*/
 | |
| 		}
 | |
| 		$this->_pref->setDataFields($dataFields)->setValidationRules($validateRules);
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set current model
 | |
| 	 *
 | |
| 	 * @return e_admin_ui
 | |
| 	 */
 | |
| 	public function _setModel()
 | |
| 	{
 | |
| 		// try to create dataFields array if missing
 | |
| 
 | |
| 		if(!$this->dataFields)
 | |
| 		{
 | |
| 			$this->dataFields = array();
 | |
| 			foreach ($this->fields as $key => $att)
 | |
| 			{
 | |
| 				if($key == $this->pid && empty($att['data'])) // Set integer as default for primary ID when not specified. MySQL Strict Fix.
 | |
| 				{
 | |
| 					$this->dataFields[$key] = 'int';
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				if((empty($att['data']) || empty($att['rule'])) && varset($att['type']) === 'comma' )
 | |
| 				{
 | |
| 					$att['data'] = 'set';
 | |
| 					$att['validate'] = 'set';
 | |
| 					$_parms = vartrue($att['writeParms'], array());
 | |
| 					if(is_string($_parms))
 | |
| 					{
 | |
| 						parse_str($_parms, $_parms);
 | |
| 					}
 | |
| 					unset($_parms['__options']);
 | |
| 					$att['rule'] = $_parms;
 | |
| 					unset($_parms);
 | |
| 				}
 | |
| 
 | |
| 				if(!empty($att['data']) && $att['data'] === 'array' && ($this->getAction() === 'inline')) // FIX for arrays being saved incorrectly with inline editing.
 | |
| 				{
 | |
| 					$att['data'] = 'set';
 | |
| 				}
 | |
| 
 | |
| 				if(!empty($att['forceSave']) || ($key !== 'options' && varset($att['data']) !== false && varset($att['type'],null) !== null && !vartrue($att['noedit'])))
 | |
| 				{
 | |
| 					$this->dataFields[$key] = vartrue($att['data'], 'str');
 | |
| 					if(!empty($att['type']))
 | |
| 					{
 | |
| 						$this->fieldInputTypes[$key] = $att['type'];
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 
 | |
| 
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// TODO - do it in one loop, or better - separate method(s) -> convertFields(validate), convertFields(data),...
 | |
| 		if(!$this->validationRules)
 | |
| 		{
 | |
| 			$this->validationRules = array();
 | |
| 			foreach ($this->fields as $key => $att)
 | |
| 			{
 | |
| 				if(varset($att['type'], null) === null || vartrue($att['noedit']))
 | |
| 				{
 | |
| 					continue;
 | |
| 				}
 | |
| 				if(!empty($att['validate']))
 | |
| 				{
 | |
| 					$this->validationRules[$key] = array(($att['validate'] === true ? 'required' : $att['validate']), varset($att['rule']), $att['title'], varset($att['error'], vartrue($att['help'])));
 | |
| 				}
 | |
| 				/*elseif(vartrue($att['check'])) could go?
 | |
| 				{
 | |
| 					$this->checkRules[$key] = array($att['check'], varset($att['rule']), $att['title'], varset($att['error'], $att['help']));
 | |
| 				}*/
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// don't touch it if already exists
 | |
| 		if($this->_model)
 | |
| 		{
 | |
| 			return $this;
 | |
| 		}
 | |
| 
 | |
| 		// default model
 | |
| 
 | |
| 
 | |
| 		$this->_model = new e_admin_model();
 | |
| 		$this->_model->setModelTable($this->table)
 | |
| 			->setFieldIdName($this->pid)
 | |
| 			->setUrl($this->url)
 | |
| 			->setValidationRules($this->validationRules)
 | |
| 			->setDbTypes($this->fieldTypes)
 | |
| 			->setFieldInputTypes($this->fieldInputTypes)
 | |
| 			->setDataFields($this->dataFields)
 | |
| 			->setMessageStackName('admin_ui_model_'.$this->table)
 | |
| 			->setParam('db_query', $this->editQry);
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set current tree
 | |
| 	 * @return e_admin_ui
 | |
| 	 */
 | |
| 	public function _setTreeModel()
 | |
| 	{
 | |
| 		// default tree model
 | |
| 		$this->_tree_model = new e_admin_tree_model();
 | |
| 		$this->_tree_model->setModelTable($this->table)
 | |
| 			->setFieldIdName($this->pid)
 | |
| 			->setUrl($this->url)
 | |
| 			->setMessageStackName('admin_ui_tree_'.$this->table)
 | |
| 			->setParams(array('model_class' => 'e_admin_model',
 | |
| 			                  'model_message_stack' => 'admin_ui_model_'.$this->table,
 | |
| 			                  'db_query' => $this->listQry,
 | |
| 			                  // Information necessary for PHP-based tree sort
 | |
| 			                  'sort_parent' => $this->getSortParent(),
 | |
| 			                  'sort_field' => $this->getSortField(),
 | |
| 			                  'primary_field' => $this->getPrimaryName(),
 | |
| 			                  ));
 | |
| 
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get extended (UI) Form instance
 | |
| 	 *
 | |
| 	 * @return e_admin_ui
 | |
| 	 */
 | |
| 	public function _setUI()
 | |
| 	{
 | |
| 		if($this->getParam('ui'))
 | |
| 		{
 | |
| 			$this->_ui = $this->getParam('ui');
 | |
| 			$this->setParam('ui', null);
 | |
| 		}
 | |
| 		else// default ui
 | |
| 		{
 | |
| 			$this->_ui = new e_admin_form_ui($this);
 | |
| 		}
 | |
| 		return $this;
 | |
| 	}
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  */
 | |
| class e_admin_form_ui extends e_form
 | |
| {
 | |
| 	/**
 | |
| 	 * @var e_admin_ui
 | |
| 	 */
 | |
| 	protected $_controller;
 | |
| 	protected $_list_view;
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor
 | |
| 	 * @param e_admin_controller_ui $controller
 | |
| 	 * @param boolean $tabindex [optional] enable form element auto tab-indexing
 | |
| 	 */
 | |
| 	public function __construct($controller, $tabindex = false)
 | |
| 	{
 | |
| 		$this->_controller = $controller;
 | |
| 		parent::__construct($tabindex);
 | |
| 
 | |
| 		// protect current methods from conflict.
 | |
| 		$this->preventConflict();
 | |
| 		// user constructor
 | |
| 		$this->init();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return void|null
 | |
| 	 */
 | |
| 	protected function preventConflict()
 | |
| 	{
 | |
| 		$err = false;
 | |
| 		$fields = $this->getController()->getFields();
 | |
| 
 | |
| 		if(empty($fields))
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		foreach($fields as $field => $foptions)
 | |
| 		{
 | |
| 			// check form custom methods
 | |
| 			if(vartrue($foptions['type']) === 'method' && method_exists('e_form', $field)) // check even if type is not method. - just in case of an upgrade later by 3rd-party.
 | |
| 			{
 | |
| 				$message = e107::getParser()->lanVars(LAN_UI_FORM_METHOD_ERROR, array('x'=>$field), true);
 | |
| 				e107::getMessage()->addError($message);
 | |
| 				$err = true;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		/*if($err)
 | |
| 		{
 | |
| 			//echo $err;
 | |
| 			//exit;
 | |
| 		}*/
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * User defined init
 | |
| 	 */
 | |
| 	public function init()
 | |
| 	{
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @todo Get a 'depth/level' field working with mysql and change the 'level' accordingly
 | |
| 	 * @param mixed $curVal
 | |
| 	 * @param string $mode read|write|inline
 | |
| 	 * @param array $parm
 | |
| 	 * @return array|string
 | |
| 	 */
 | |
| 	public function treePrefix($curVal, $mode, $parm)
 | |
| 	{
 | |
| 		$controller         = $this->getController();
 | |
| 		$parentField        = $controller->getSortParent();
 | |
| 		$treePrefixField    = $controller->getTreePrefix();
 | |
| 		$parent 	        = $controller->getListModel()->get($parentField);
 | |
| 		$level              = $controller->getListModel()->get('_depth');
 | |
| 
 | |
| 
 | |
| 		if($mode === 'read')
 | |
| 		{
 | |
| 
 | |
| 			$inline = $this->getController()->getFieldAttr($treePrefixField,'inline');
 | |
| 
 | |
| 			if($inline === true)
 | |
| 			{
 | |
| 				return $curVal;
 | |
| 			}
 | |
| 
 | |
| 			$level_image = $parent ? str_replace('level-x','level-'.$level, defset('ADMIN_CHILD_ICON')) : '';
 | |
| 
 | |
| 			return ($parent) ?  $level_image.$curVal : $curVal;
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		if($mode === 'inline')
 | |
| 		{
 | |
| 			$ret = array('inlineType'=>'text');
 | |
| 
 | |
| 			if(!empty($parent))
 | |
| 			{
 | |
| 				$ret['inlineParms'] = array('pre'=> str_replace('level-x','level-'.$level, defset('ADMIN_CHILD_ICON')));
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 			return $ret;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| /*
 | |
| 			if($mode == 'write') //  not used.
 | |
| 			{
 | |
| 			//	return $frm->text('forum_name',$curVal,255,'size=xxlarge');
 | |
| 			}
 | |
| 
 | |
| 			if($mode == 'filter')
 | |
| 			{
 | |
| 				return;
 | |
| 			}
 | |
| 			if($mode == 'batch')
 | |
| 			{
 | |
| 				return;
 | |
| 			}
 | |
| */
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic DB Record Creation Form.
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getCreate()
 | |
| 	{
 | |
| 		$controller = $this->getController();
 | |
| 		$request = $controller->getRequest();
 | |
| 		if($controller->getId())
 | |
| 		{
 | |
| 			$legend = e107::getParser()->lanVars(LAN_UI_EDIT_LABEL, $controller->getId()); // sprintXXX(LAN_UI_EDIT_LABEL, $controller->getId());
 | |
| 			$form_start = vartrue($controller->headerUpdateMarkup);
 | |
| 			$form_end = vartrue($controller->footerUpdateMarkup);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$legend = LAN_UI_CREATE_LABEL;
 | |
| 			$form_start = vartrue($controller->headerCreateMarkup);
 | |
| 			$form_end = vartrue($controller->footerCreateMarkup);
 | |
| 		}
 | |
| 
 | |
| 		$tabs = $controller->getTabs();
 | |
| 
 | |
| 		if($multiLangInfo = $this->renderLanguageTableInfo())
 | |
| 		{
 | |
| 			if(empty($tabs))
 | |
| 			{
 | |
| 				$head = "<div id='admin-ui-edit-db-language' class='text-right'>".$multiLangInfo. '</div>';
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$head = "<div id='admin-ui-edit-db-language' class='text-right tabs'>".$multiLangInfo. '</div>';
 | |
| 			}
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$head = '';
 | |
| 		}
 | |
| 
 | |
| 		$models = array();
 | |
| 		$forms = array();
 | |
| 		$forms[] = array(
 | |
| 				'id'  => $this->getElementId(),
 | |
| 				'header' => $head,
 | |
| 				'footer' => '',
 | |
| 				//'url' => e_SELF,
 | |
| 				//'query' => 'self', or custom GET query, self is default
 | |
| 				'fieldsets' => array(
 | |
| 					'create' => array(
 | |
| 						'tabs'	=>  $tabs, //used within a single form.
 | |
| 						'legend' => $legend,
 | |
| 						'fields' => $controller->getFields(), //see e_admin_ui::$fields
 | |
| 						'header' => $form_start, //XXX Unused?
 | |
| 						'footer' => $form_end,  //XXX Unused?
 | |
| 						'after_submit_options' => $controller->getAfterSubmitOptions(), // or true for default redirect options
 | |
| 						'after_submit_default' => $request->getPosted('__after_submit_action', $controller->getDefaultAction()), // or true for default redirect options
 | |
| 						'triggers' => $controller->getDefaultTrigger(), // standard create/update-cancel triggers
 | |
| 					)
 | |
| 				)
 | |
| 		);
 | |
| 
 | |
| 		$models[] = $controller->getModel();
 | |
| 
 | |
| 		return $this->renderCreateForm($forms, $models, e_AJAX_REQUEST);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generic Settings Form.
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getSettings()
 | |
| 	{
 | |
| 		$controller = $this->getController();
 | |
| 	//	$request = $controller->getRequest();
 | |
| 		$legend = LAN_UI_PREF_LABEL;
 | |
| 		$models = array();
 | |
| 		$forms = array();
 | |
| 		$forms[] = array(
 | |
| 				'id'  => $this->getElementId(),
 | |
| 				//'url' => e_SELF,
 | |
| 				//'query' => 'self', or custom GET query, self is default
 | |
| 				'tabs' => false, // TODO - NOT IMPLEMENTED YET - enable tabs (only if fieldset count is > 1)
 | |
| 				'fieldsets' => array(
 | |
| 					'settings' => array(
 | |
| 						'tabs'	=> $controller->getPrefTabs(), //used within a single form. 
 | |
| 						'legend' => $legend,
 | |
| 						'fields' => $controller->getPrefs(), //see e_admin_ui::$prefs
 | |
| 						'after_submit_options' => false,
 | |
| 						'after_submit_default' => false, // or true for default redirect options
 | |
| 						'triggers' => array('save' => array(LAN_SAVE, 'update')), // standard create/update-cancel triggers
 | |
| 					)
 | |
| 				)
 | |
| 		);
 | |
| 		$models[] = $controller->getConfig();
 | |
| 
 | |
| 	//	print_a($forms);
 | |
| 
 | |
| 		return $this->renderCreateForm($forms, $models, e_AJAX_REQUEST);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Integrate e_addon data into the list model.
 | |
| 	 * @param e_tree_model $tree
 | |
| 	 * @param array $fields
 | |
| 	 * @param string $pid
 | |
| 	 * @return null
 | |
| 	 */
 | |
| 	private function setAdminAddonModel(e_tree_model $tree, $fields, $pid)
 | |
| 	{
 | |
| 
 | |
| 		$event= $this->getController()->getEventName();
 | |
| 
 | |
| 		$arr = array();
 | |
| 
 | |
| 		/** @var e_tree_model $model */
 | |
| 		foreach($tree->getTree() as $model)
 | |
| 		{
 | |
| 			foreach($fields as $fld)
 | |
| 			{
 | |
| 
 | |
| 				if(strpos($fld,'x_') !== 0)
 | |
| 				{
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				list($prefix,$plug,$field) = explode('_',$fld,3);
 | |
| 
 | |
| 				if($prefix !== 'x' || empty($field) || empty($plug))
 | |
| 				{
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				$id = $model->get($pid);
 | |
| 
 | |
| 				if(!empty($id))
 | |
| 				{
 | |
| 					$arr[$plug][$field][$id] = $model;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		foreach($arr as $plug=>$field)
 | |
| 		{
 | |
| 
 | |
| 			if($obj = e107::getAddon($plug, 'e_admin'))
 | |
| 			{
 | |
| 				foreach($field as $fld=>$var)
 | |
| 				{
 | |
| 					$ids = implode(',', array_keys($var));
 | |
| 
 | |
| 					$value = (array) e107::callMethod($obj,'load', $event,$ids);
 | |
| 
 | |
| 				//	$value = (array) $obj->load($event, $ids);
 | |
| 
 | |
| 					foreach($var as $id=>$model)
 | |
| 					{
 | |
| 						$model->set('x_' .$plug. '_' .$fld, varset($value[$id][$fld],null));
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Create list view
 | |
| 	 * Search for the following GET variables:
 | |
| 	 * - from: integer, current page
 | |
| 	 *
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getList($ajax = false, $view='default')
 | |
| 	{
 | |
| 		$tp = e107::getParser();
 | |
| 		$this->_list_view = $view;
 | |
| 		$controller = $this->getController();
 | |
| 
 | |
| 		$request = $controller->getRequest();
 | |
| 		$id = $this->getElementId();
 | |
| 		$pid = $controller->getPrimaryName();
 | |
| 		$options = array();
 | |
| 		$tree = array();
 | |
| 		$tree[$id] = $controller->getTreeModel();
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 		if($view === 'default' && deftrue('e_DEBUG_TREESORT'))
 | |
| 		{
 | |
| 			$controller->getTreeModelSorted();
 | |
| 		}
 | |
| 
 | |
| 		// if going through confirm screen - no JS confirm
 | |
| 		$controller->setFieldAttr('options', 'noConfirm', $controller->deleteConfirmScreen);
 | |
| 
 | |
| 		$fields = $controller->getFields();
 | |
| 
 | |
| 		$this->setAdminAddonModel($tree[$id], array_keys($fields), $pid);
 | |
| 
 | |
| 		// checks dispatcher acess/perms for create/edit/delete access in list mode.
 | |
| 		$mode           = $controller->getMode();
 | |
| 		$deleteRoute    = $mode. '/delete';
 | |
| 		$editRoute      = $mode. '/edit';
 | |
| 		$createRoute    = $mode. '/create';
 | |
| 
 | |
| 		if(!$controller->getDispatcher()->hasRouteAccess($createRoute)) // disable the batchCopy option.
 | |
| 		{
 | |
| 			$controller->setBatchCopy(false);
 | |
| 		}
 | |
| 
 | |
| 		if(!$controller->getDispatcher()->hasRouteAccess($deleteRoute)) // disable the delete button and batch delete.
 | |
| 		{
 | |
| 			$fields['options']['readParms']['deleteClass'] = e_UC_NOBODY;
 | |
| 			$controller->setBatchDelete(false);
 | |
| 		}
 | |
| 
 | |
| 		if(!$controller->getDispatcher()->hasRouteAccess($editRoute))
 | |
| 		{
 | |
| 			$fields['options']['readParms']['editClass'] = e_UC_NOBODY; // display the edit button.
 | |
| 			foreach($options[$id]['fields'] as $k=>$v) // disable inline editing.
 | |
| 			{
 | |
| 				$fields[$k]['inline'] = false;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if(!$controller->getSortField())
 | |
| 		{
 | |
| 			$fields['options']['sort'] = false;
 | |
| 		}
 | |
| 
 | |
| 		if($treefld = $controller->getTreePrefix())
 | |
| 		{
 | |
| 			$fields[$treefld]['type'] = 'method';
 | |
| 			$fields[$treefld]['method'] = 'treePrefix'; /* @see e_admin_form_ui::treePrefix(); */
 | |
| 
 | |
| 			$tr = $controller->getTreeModel()->toArray();
 | |
| 
 | |
| 			foreach($tr as $row)
 | |
| 			{
 | |
| 				e107::getDebug()->log($row[$treefld].' >  '.$row['_treesort']);
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 		// ------------------------------------------
 | |
| 
 | |
| 		$coreBatchOptions = array(
 | |
| 			'delete'        => $controller->getBatchDelete(),
 | |
| 			'copy'          => $controller->getBatchCopy(),
 | |
| 			'url'           => $controller->getBatchLink(),
 | |
| 			'featurebox'    => $controller->getBatchFeaturebox(),
 | |
| 			'export'        => $controller->getBatchExport(),
 | |
| 
 | |
| 		);
 | |
| 
 | |
| 
 | |
| 		$options[$id] = array(
 | |
| 			'id'            => $this->getElementId(), // unique string used for building element ids, REQUIRED
 | |
| 			'pid'           => $pid, // primary field name, REQUIRED
 | |
| 			'query'	        => $controller->getFormQuery(), // work around - see form in newspost.php (submitted news)
 | |
| 			'head_query'    => $request->buildQueryString('field=[FIELD]&asc=[ASC]&from=[FROM]', false), // without field, asc and from vars, REQUIRED
 | |
| 			'np_query'      => $request->buildQueryString(array(), false, 'from'), // without from var, REQUIRED for next/prev functionality
 | |
| 			'legend'        => $controller->getPluginTitle(), // hidden by default
 | |
| 			'form_pre'      => !$ajax ? $this->renderFilter(array($controller->getQuery('searchquery'), $controller->getQuery('filter_options')), $controller->getMode().'/'.$controller->getAction()) : '', // needs to be visible when a search returns nothing
 | |
| 			'form_post'     => '', // markup to be added after closing form element
 | |
| 			'fields'        => $fields, // see e_admin_ui::$fields
 | |
| 			'fieldpref'     => $controller->getFieldPref(), // see e_admin_ui::$fieldpref
 | |
| 			'table_pre'     => '', // markup to be added before opening table element
 | |
| 		//	'table_post' => !$tree[$id]->isEmpty() ? $this->renderBatch($controller->getBatchDelete(),$controller->getBatchCopy(),$controller->getBatchLink(),$controller->getBatchFeaturebox()) : '',
 | |
| 
 | |
| 			'table_post'    => $this->renderBatch($coreBatchOptions, $controller->getBatchOptions()),
 | |
| 			'fieldset_pre'  => '', // markup to be added before opening fieldset element
 | |
| 			'fieldset_post' => '', // markup to be added after closing fieldset element
 | |
| 			'grid'          =>  $controller->getGrid(),
 | |
| 			'perPage'       => $controller->getPerPage(), // if 0 - no next/prev navigation
 | |
| 			'from'          => $controller->getQuery('from', 0), // current page, default 0
 | |
| 			'field'         => $controller->getQuery('field'), //current order field name, default - primary field
 | |
| 			'asc'           => $controller->getQuery('asc', 'desc'), //current 'order by' rule, default 'asc'
 | |
| 		);
 | |
| 
 | |
| 
 | |
| 
 | |
| 		if($view === 'grid')
 | |
| 		{
 | |
| 			return $this->renderGridForm($options, $tree, $ajax);
 | |
| 		}
 | |
| 
 | |
| 		return $this->renderListForm($options, $tree, $ajax);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $ids
 | |
| 	 * @param $ajax
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getConfirmDelete($ids, $ajax = false)
 | |
| 	{
 | |
| 		$controller = $this->getController();
 | |
| 		$request = $controller->getRequest();
 | |
| 		$fieldsets = array();
 | |
| 		$forms = array();
 | |
| 		$id_array = explode(',', $ids);
 | |
| 		$delcount = count($id_array);
 | |
| 		
 | |
| 		if(!empty($controller->deleteConfirmMessage))
 | |
|         { 
 | |
| 			e107::getMessage()->addWarning(str_replace('[x]', '<b>' .$delcount. '</b>', $controller->deleteConfirmMessage));
 | |
|         }
 | |
| 		else 
 | |
|         {
 | |
| 			e107::getMessage()->addWarning(str_replace('[x]', '<b>' .$delcount. '</b>',LAN_UI_DELETE_WARNING));
 | |
|         }
 | |
|     
 | |
| 		$fieldsets['confirm'] = array(
 | |
| 			'fieldset_pre' => '', // markup to be added before opening fieldset element
 | |
| 			'fieldset_post' => '', // markup to be added after closing fieldset element
 | |
| 			'table_head' => '', // markup between <thead> tag
 | |
| 			// Colgroup Example: array(0 => array('class' => 'label', 'style' => 'text-align: left'), 1 => array('class' => 'control', 'style' => 'text-align: left'));
 | |
| 			'table_colgroup' => '', // array to be used for creating markup between  <colgroup> tag (<col> list)
 | |
| 			'table_pre' => '', // markup to be added before opening table element
 | |
| 			'table_post' => '', // markup to be added after closing table element
 | |
| 			'table_rows' => '', // rows array (<td> tags)
 | |
| 			'table_body' => '', // string body - used only if rows empty
 | |
| 			'pre_triggers' => '',
 | |
| 			'triggers' => array('hidden' => $this->hidden('etrigger_delete['.$ids.']', $ids) . $this->token(), 'delete_confirm' => array(LAN_CONFDELETE, 'confirm', $ids), 'cancel' => array(LAN_CANCEL, 'cancel')),
 | |
| 		);
 | |
| 		if($delcount > 1)
 | |
| 		{
 | |
| 			$fieldsets['confirm']['triggers']['hidden'] = $this->hidden('etrigger_batch', 'delete');
 | |
| 		}
 | |
| 
 | |
| 		$id = null;
 | |
| 		$forms[$id] = array(
 | |
| 			'id' => $this->getElementId(), // unique string used for building element ids, REQUIRED
 | |
| 			'url' => e_REQUEST_SELF, // default
 | |
| 			'query' => $request->buildQueryString(array(), true, 'ajax_used'), // - ajax_used is now removed from QUERY_STRING - class2
 | |
| 			'legend' => $controller->addTitle(LAN_UI_DELETE_LABEL), // hidden by default
 | |
| 			'form_pre' => '',  // markup to be added before opening form element
 | |
| 			'form_post' => '', // markup to be added after closing form element
 | |
| 			'header' => '', // markup to be added after opening form element
 | |
| 			'footer' => '', // markup to be added before closing form element
 | |
| 			'fieldsets' => $fieldsets,
 | |
| 		);
 | |
| 		return $this->renderForm($forms, $ajax);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Render pagination
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function renderPagination()
 | |
| 	{
 | |
| 		if($this->_list_view === 'grid' && $this->getController()->getGrid('carousel') === true)
 | |
| 		{
 | |
| 			return '<div class="btn-group" >
 | |
| 			<a id="admin-ui-carousel-prev" class="btn btn-default btn-secondary" href="#admin-ui-carousel" data-slide="prev"><i class="fa fa-backward"></i></a>
 | |
| 			<a id="admin-ui-carousel-index" class="btn btn-default btn-secondary">1</a>
 | |
| 			<a id="admin-ui-carousel-next" class="btn btn-default btn-secondary" href="#admin-ui-carousel" data-slide="next"><i class="fa fa-forward"></i></a>
 | |
| 			</div>';
 | |
| 		}
 | |
| 
 | |
| 		$tree           = $this->getController()->getTreeModel();
 | |
| 		$totalRecords   = $tree->getTotal();
 | |
| 		$perPage        = $this->getController()->getPerPage();
 | |
| 		$fromPage       = $this->getController()->getQuery('from', 0);
 | |
| 
 | |
| 		$vars           = $this->getController()->getQuery();
 | |
| 		$vars['from']   = '[FROM]';
 | |
| 
 | |
| 		$paginate       = http_build_query($vars, '', '&');
 | |
| 
 | |
| 		e107::js('footer-inline', "
 | |
| 				\$('#admin-ui-list-filter a.nextprev-item').on('click', function() {
 | |
| 					\$('#admin-ui-list-filter .indicator').show();
 | |
| 			});
 | |
| 		
 | |
| 		");
 | |
| 
 | |
| 		return $this->pagination(e_REQUEST_SELF.'?'.$paginate,$totalRecords,$fromPage,$perPage,array('template'=>'basic'));
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param $current_query
 | |
| 	 * @param $location
 | |
| 	 * @param $input_options
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function renderFilter($current_query = array(), $location = '', $input_options = array())
 | |
| 	{
 | |
| 
 | |
| 		if(!$input_options)
 | |
| 		{
 | |
| 			$input_options = array('size' => 20);
 | |
| 		}
 | |
| 		if(!$location)
 | |
| 		{
 | |
| 			$location = 'main/list'; //default location
 | |
| 		}
 | |
| 		$l = e107::getParser()->post_toForm(explode('/', $location));
 | |
| 		if(!is_array($input_options))
 | |
| 		{
 | |
| 			parse_str($input_options, $input_options);
 | |
| 		}
 | |
| 		$input_options['id'] = false;
 | |
| 		$input_options['class'] = 'tbox input-text filter input-xlarge ';
 | |
| 		$controller = $this->getController();
 | |
| 		$filter_pre = vartrue($controller->preFilterMarkup);
 | |
| 		$filter_post = vartrue($controller->postFilterMarkup);
 | |
| 		$filter_preserve_var = array();
 | |
| 		// method requires controller - stanalone advanced usage not possible 
 | |
| 		if($this->getController())
 | |
| 		{
 | |
| 			$get = $this->getController()->getQuery();
 | |
| 			foreach ($get as $key => $value) 
 | |
| 			{
 | |
| 				if($key === 'searchquery' || $key === 'filter_options' || $key === 'etrigger_filter')
 | |
| 				{
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				// Reset pager after filtering.
 | |
| 				if ($key === 'from')
 | |
| 				{
 | |
| 					continue;
 | |
| 				}
 | |
| 				
 | |
| 				$key = preg_replace('/[\W]/', '', $key);
 | |
| 				$filter_preserve_var[] = $this->hidden($key, rawurlencode($value));
 | |
| 			}
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$filter_preserve_var[] = $this->hidden('mode', $l[0]);
 | |
| 			$filter_preserve_var[] = $this->hidden('action', $l[1]);
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		//	$tree = $this->getTree();
 | |
| 		//	$total = $this->getTotal();
 | |
| 		$grid = $this->getController()->getGrid();
 | |
| 
 | |
| 
 | |
| 		$gridToggle = '';
 | |
| 
 | |
| 		if(!empty($grid) && varset($grid['toggleButton']) !==false)
 | |
| 		{
 | |
| 			$gridAction = $this->getController()->getAction() === 'grid' ? 'list' : 'grid';
 | |
| 			$gridQuery = (array) $_GET;
 | |
| 			$gridQuery['action'] = $gridAction;
 | |
| 			$toggleUrl = e_REQUEST_SELF. '?' .http_build_query($gridQuery, '', '&');
 | |
| 			$gridIcon = ($gridAction === 'grid') ? defset('ADMIN_GRID_ICON') : defset('ADMIN_LIST_ICON');
 | |
| 			$gridTitle = ($gridAction === 'grid') ? LAN_UI_VIEW_GRID_LABEL : LAN_UI_VIEW_LIST_LABEL;
 | |
| 			$gridToggle = "<a class='btn btn-default' href='".$toggleUrl."' title=\"".$gridTitle. '">' .$gridIcon. '</a>';
 | |
| 		}
 | |
| 
 | |
| 	// <!--<i class='fa fa-search searchquery form-control-feedback form-control-feedback-left'></i>-->
 | |
| 
 | |
| 		$text = "
 | |
| 			<form method='get' action='".e_REQUEST_SELF."'>
 | |
| 				<fieldset id='admin-ui-list-filter' class='e-filter'>
 | |
| 					<legend class='e-hideme'>".LAN_LABEL_LABEL_SELECTED. '</legend>
 | |
| 					' .$filter_pre."
 | |
| 					<div class='row-fluid'>
 | |
| 						<div  class='left form-inline span8 col-md-8' >
 | |
| 							<span id='admin-ui-list-search' class='form-group has-feedback has-feedback-left'>
 | |
| 								".$this->text('searchquery', $current_query[0], 50, $input_options). '					
 | |
| 							</span>
 | |
| 							' .$this->select_open('filter_options', array('class' => 'form-control e-tip tbox select filter', 'id' => false, 'title' =>LAN_FILTER)). '
 | |
| 								' .$this->option(LAN_FILTER_LABEL_DISPLAYALL, ''). '
 | |
| 								' .$this->option(LAN_FILTER_LABEL_CLEAR, '___reset___'). '
 | |
| 								' .$this->renderBatchFilter('filter', $current_query[1]). '
 | |
| 							' .$this->select_close()."
 | |
| 							<div class='e-autocomplete'></div>
 | |
| 							".implode("\n", $filter_preserve_var). '
 | |
| 							' .$this->admin_button('etrigger_filter', 'etrigger_filter', 'filter e-hide-if-js', defset('ADMIN_FILTER_ICON'), array('id' => false, 'title' =>LAN_FILTER, 'loading' => false)). '
 | |
| 							
 | |
| 							' .$this->renderPagination()."	
 | |
| 							".$gridToggle."
 | |
| 							<span class='indicator' style='display: none;'>
 | |
| 								<i class='fa fa-spin fa-spinner fa-fw'></i>
 | |
| 							</span>
 | |
| 						</div>
 | |
| 						<div class='span4 col-md-4 text-right' >";
 | |
| 
 | |
| 						// Let Admin know which language table is being saved to. (avoid default table overwrites) 
 | |
| 						if($languageInfo = $this->renderLanguageTableInfo())
 | |
| 						{
 | |
| 							$text .= "<div id='admin-ui-list-db-language' >".$languageInfo.'</div>';
 | |
| 						}
 | |
| 
 | |
| 						$text .= $this->renderCustomListButton(); // Optional
 | |
| 
 | |
| 						$text .= '
 | |
| 						</div>
 | |
| 					</div>
 | |
| 					' .$filter_post. '
 | |
| 				</fieldset>
 | |
| 			</form>
 | |
| 		';
 | |
| 
 | |
| 	
 | |
| 		e107::js('core','scriptaculous/controls.js','prototype', 2);
 | |
| 		//TODO - external JS
 | |
| 		e107::js('footer-inline',"
 | |
| 	
 | |
| 	            //autocomplete fields
 | |
| 	             \$\$('input[name=searchquery]').each(function(el, cnt) {
 | |
| 				 	if(!cnt) el.activate();
 | |
| 					else return;
 | |
| 					new Ajax.Autocompleter(el, el.next('div.e-autocomplete'), '".e_REQUEST_SELF. '?mode=' .$l[0]."&action=filter', {
 | |
| 					  paramName: 'searchquery',
 | |
| 					  minChars: 2,
 | |
| 					  frequency: 0.5,
 | |
| 					  afterUpdateElement: function(txt, li) {
 | |
| 					  	var cfrm = el.up('form'), cont = cfrm.next('.e-container');
 | |
| 						if(!cont) {
 | |
| 							return;
 | |
| 						}
 | |
| 					  	cfrm.submitForm(cont);
 | |
| 					  },
 | |
| 					  indicator:  el.next('span.indicator'),
 | |
| 					  parameters: 'ajax_used=1'
 | |
| 					});
 | |
| 					var sel = el.next('select.filter');
 | |
| 					if(sel) {
 | |
| 						sel.observe('change', function (e) {
 | |
| 							var cfrm = e.element().up('form'), cont = cfrm.next('.e-container');
 | |
| 							if(cfrm && cont && e.element().value != '___reset___') {
 | |
| 								e.stop();
 | |
| 								cfrm.submitForm(cont);
 | |
| 								return;
 | |
| 							}
 | |
| 							e107Helper.selectAutoSubmit(e.element());
 | |
| 						});
 | |
| 					}
 | |
| 				});
 | |
| 		",'prototype');
 | |
| 		
 | |
| 		// TODO implement ajax queue
 | |
| 		// FIXME
 | |
| 		// dirty way to register events after ajax update - DO IT RIGHT - see all.jquery, create object and use handler,
 | |
| 		// re-register them global after ajax update (context)... use behaviors and call e107.attachBehaviors();
 | |
| 		e107::js('footer-inline',"
 | |
| 			var filterRunning = false, request;
 | |
| 			var applyAfterAjax = function(context) {
 | |
| 		      	\$('.e-expandit', context).click(function () {
 | |
| 		       		var href = (\$(this).is('a')) ? \$(this).attr('href') : '';
 | |
| 		       		if(href == '' && \$(this).attr('data-target'))
 | |
| 		       		{
 | |
| 		       			href = '#' + \$(this).attr('data-target');	
 | |
| 		       		}
 | |
| 					if(href === '#' || href == '') 
 | |
| 					{
 | |
| 						idt = \$(this).nextAll('div');	
 | |
| 						\$(idt).toggle('slow');
 | |
| 						 return true;			
 | |
| 					}
 | |
| 		       		//var id = $(this).attr('href');   		
 | |
| 					\$(href).toggle('slow');
 | |
| 					return false;
 | |
| 				}); 
 | |
| 				\$('input.toggle-all', context).click(function(evt) {
 | |
| 					var selector = 'input[type=\"checkbox\"].checkbox';
 | |
| 					if(\$(this).val().startsWith('jstarget:')) {
 | |
| 						selector = 'input[type=\"checkbox\"][name^=\"' + \$(this).val().split(/jstarget\:/)[1] + '\"]';
 | |
| 					}
 | |
| 					
 | |
| 					if(\$(this).is(':checked')){
 | |
| 						\$(selector).attr('checked', 'checked');
 | |
| 					}
 | |
| 					else{
 | |
| 						\$(selector).removeAttr('checked');
 | |
| 					}
 | |
| 				});
 | |
| 			};
 | |
| 			var searchQueryHandler = function (e) {
 | |
| 				
 | |
| 				var el = \$(this), frm = el.parents('form'), cont = frm.nextAll('.e-container');
 | |
| 				if(cont.length < 1 || frm.length < 1 || (el.val().length > 0 && el.val().length < 3)) return;
 | |
| 				e.preventDefault();
 | |
| 				
 | |
| 				if(filterRunning && request) request.abort();
 | |
| 				filterRunning = true;
 | |
| 				\$('#admin-ui-list-filter .indicator').show();
 | |
| 				cont.css({ opacity: 0.5 });
 | |
| 				
 | |
| 				request = \$.get(frm.attr('action'), frm.serialize(), function(data){
 | |
| 					filterRunning = false;
 | |
| 					setTimeout(function() {
 | |
| 						if(filterRunning) {
 | |
| 							//cont.css({ opacity: 1 });
 | |
| 							return;
 | |
| 						}
 | |
| 						cont.html(data).css({ opacity: 1 });
 | |
| 						\$('#admin-ui-list-filter .indicator').hide();
 | |
| 						// TODO remove applyAfterAjax() and use behaviors!
 | |
| 						applyAfterAjax(cont);
 | |
| 						// Attach behaviors to the newly loaded contents.
 | |
| 						e107.attachBehaviors();
 | |
| 					}, 700);
 | |
| 				}, 'html')
 | |
| 				.error(function() {
 | |
| 					filterRunning = false;
 | |
| 					cont.css({ opacity: 1 });
 | |
| 				});
 | |
| 			};
 | |
| 			\$('#searchquery').on('keyup', searchQueryHandler);
 | |
| 			
 | |
| 			\$('#filter-options').on('change', function() {
 | |
| 					\$('#admin-ui-list-filter .indicator').show();
 | |
| 			});
 | |
| 			
 | |
| 			\$('#etrigger-filter').on('click', function() {
 | |
| 					\$('#admin-ui-list-filter .indicator').show();
 | |
| 			});
 | |
| 			
 | |
| 			
 | |
| 		", 'jquery');
 | |
| 
 | |
| 		return $text;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Optional
 | |
| 	 * @return null|string
 | |
| 	 */
 | |
| 	public function renderCustomListButton()
 | |
| 	{
 | |
| 		return null; // "<a class='btn btn-primary' href=''>Add New</a>";
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string|null
 | |
| 	 */
 | |
| 	private function renderLanguageTableInfo()
 | |
| 	{
 | |
| 
 | |
| 		if(!e107::getConfig()->get('multilanguage'))
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		$curTable = $this->getController()->getTableName();
 | |
| 		$sitelanguage = e107::getConfig()->get('sitelanguage');
 | |
| 
 | |
| 		$val = e107::getDb()->hasLanguage($curTable, true);
 | |
| 
 | |
| 		if($val === false)
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 
 | |
| 		if($curTable != e107::getDb()->hasLanguage($curTable))
 | |
| 		{
 | |
| 			$lang = e107::getDb()->mySQLlanguage;
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$lang = $sitelanguage;
 | |
| 		}
 | |
| 
 | |
| 		$def = deftrue('LAN_UI_USING_DATABASE_TABLE','Using [x] database table');
 | |
| 		$diz  = e107::getParser()->lanVars($def, $lang); // "Using ".$lang." database table";
 | |
| 		$class = ($sitelanguage == $lang) ? 'default' : '';
 | |
| 
 | |
| 		$text = "<span class='adminui-language-table-info ".$class." e-tip' title=\"".$diz. '">';
 | |
| 		$text .= e107::getParser()->toGlyph('fa-hdd-o'); // '<i class="icon-hdd"></i> ';
 | |
| 		$text .= e107::getLanguage()->toNative($lang). '</span>';
 | |
| 		return $text;
 | |
| 
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 
 | |
| 	// FIXME - use e_form::batchoptions(), nice way of building batch dropdown - news administration show_batch_options()
 | |
| 
 | |
| 	/**
 | |
| 	 * @param array $options array of flags for copy, delete, url, featurebox, batch
 | |
| 	 * @param array $customBatchOptions
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function renderBatch($options, $customBatchOptions=array())
 | |
| 	{
 | |
| 
 | |
| 		$fields = $this->getController()->getFields();
 | |
| 
 | |
| 		if(!varset($fields['checkboxes']))
 | |
| 		{
 | |
| 			$mes = e107::getMessage();
 | |
| 			$mes->add("Cannot display Batch drop-down as 'checkboxes' was not found in \$fields array.", E_MESSAGE_DEBUG);
 | |
| 			return '';
 | |
| 		}
 | |
| 		
 | |
| 		// FIX - don't show FB option if plugin not installed
 | |
| 		if(!e107::isInstalled('featurebox'))
 | |
| 		{
 | |
| 			$options['featurebox'] = false;
 | |
| 		}
 | |
| 		
 | |
| 		// TODO - core ui-batch-option class!!! REMOVE INLINE STYLE!
 | |
| 		// XXX Quick Fix for styling - correct. 
 | |
| 		$text = "
 | |
| 			<div id='admin-ui-list-batch' class='navbar navbar-inner left' >
 | |
| 				<div class='span6 col-md-6'>";
 | |
| 
 | |
| 		$selectStart = "<div class='form-inline input-inline'>
 | |
| 					".defset('ADMIN_CHILD_ICON')."
 | |
| 	         		 		<div class='input-group input-append'>
 | |
| 						".$this->select_open('etrigger_batch', array('class' => 'tbox form-control input-large select batch e-autosubmit reset', 'id' => false)). '
 | |
| 						' .$this->option(defset('LAN_BATCH_LABEL_SELECTED'), '');
 | |
| 
 | |
| 		$selectOpt = '';
 | |
| 				
 | |
| 		if(!$this->getController()->getTreeModel()->isEmpty())
 | |
| 		{		
 | |
| 			$selectOpt .= !empty($options['copy']) ? $this->option(defset('LAN_COPY'),
 | |
| 				'copy',
 | |
| 				false,
 | |
| 				array('class' => 'ui-batch-option class', 'other' => 'style="padding-left: 15px"')) : '';
 | |
| 			$selectOpt .= !empty($options['delete']) ? $this->option(defset('LAN_DELETE'),
 | |
| 				'delete',
 | |
| 				false,
 | |
| 				array('class' => 'ui-batch-option class', 'other' => 'style="padding-left: 15px"')) : '';
 | |
| 			$selectOpt .= !empty($options['export']) ? $this->option(defset('LAN_UI_BATCH_EXPORT'),
 | |
| 				'export',
 | |
| 				false,
 | |
| 				array('class' => 'ui-batch-option class', 'other' => 'style="padding-left: 15px"')) : '';
 | |
| 			$selectOpt .= !empty($options['url']) ? $this->option(defset('LAN_UI_BATCH_CREATELINK'),
 | |
| 				'url',
 | |
| 				false,
 | |
| 				array('class' => 'ui-batch-option class', 'other' => 'style="padding-left: 15px"')) : '';
 | |
| 			$selectOpt .= !empty($options['featurebox']) ? $this->option(defset('LAN_PLUGIN_FEATUREBOX_BATCH'),
 | |
| 				'featurebox',
 | |
| 				false,
 | |
| 				array('class' => 'ui-batch-option class', 'other' => 'style="padding-left: 15px"')) : '';
 | |
| 
 | |
| 		//	if(!empty($parms['sef'])
 | |
| 
 | |
| 
 | |
| 
 | |
| 			if(!empty($customBatchOptions))
 | |
| 			{
 | |
| 				foreach($customBatchOptions as $key=>$val)
 | |
| 				{
 | |
| 
 | |
| 					if(is_array($val))
 | |
| 					{
 | |
| 						$selectOpt .= $this->optgroup_open($key);
 | |
| 
 | |
| 						foreach($val as $k=>$v)
 | |
| 						{
 | |
| 							$selectOpt .= $this->option($v, $k, false, array('class' => 'ui-batch-option class', 'other' => 'style="padding-left: 15px"'));
 | |
| 						}
 | |
| 
 | |
| 						$selectOpt .= $this->optgroup_close();
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						$selectOpt .= $this->option($val, $key, false, array('class' => 'ui-batch-option class', 'other' => 'style="padding-left: 15px"'));
 | |
| 					}
 | |
| 
 | |
| 
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 			$selectOpt .= $this->renderBatchFilter();
 | |
| 
 | |
| 			if(!empty($selectOpt))
 | |
| 			{
 | |
| 				$text .= $selectStart;
 | |
| 
 | |
| 				$text .= $selectOpt;
 | |
| 
 | |
| 				$text .= $this->select_close();
 | |
| 
 | |
| 				$text .= "<div class='input-group-btn input-append'>
 | |
| 				".$this->admin_button('e__execute_batch', 'e__execute_batch', 'batch e-hide-if-js', LAN_GO, array('id' => false)). '
 | |
| 				</div>';
 | |
| 				$text .= '</div></div>';
 | |
| 			}
 | |
| 
 | |
| 			$text .= '</div>';
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		
 | |
| 		$text .= "
 | |
| 
 | |
| 				<div id='admin-ui-list-total-records' class='span6 col-md-6 right'><span>".e107::getParser()->lanVars(LAN_UI_TOTAL_RECORDS,number_format($this->getController()->getTreeModel()->getTotal())). '</span></div>
 | |
| 			</div>
 | |
| 		';
 | |
| 
 | |
| 
 | |
| 		return $text;
 | |
| 	}
 | |
| 
 | |
| 	private function sortFieldsByPriority(array $fields, array $priorityKeys)
 | |
| 	{
 | |
| 	    // Separate fields into priority and non-priority arrays
 | |
| 	    $priorityFields = [];
 | |
| 	    $remainingFields = $fields;
 | |
| 
 | |
| 	    foreach ($priorityKeys as $key)
 | |
| 	    {
 | |
| 	        // If the key exists in the fields array, move it to the priorityFields array
 | |
| 	        if (isset($fields[$key]))
 | |
| 	        {
 | |
| 	            $priorityFields[$key] = $fields[$key];
 | |
| 	            unset($remainingFields[$key]); // Remove from remaining fields
 | |
| 	        }
 | |
| 	    }
 | |
| 
 | |
| 	    // Merge priorityFields at the top with remainingFields retaining their original order
 | |
| 	    return $priorityFields + $remainingFields;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Render Batch and Filter Dropdown options.
 | |
| 	 * @param string $type
 | |
| 	 * @param string $selected
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function renderBatchFilter($type='batch', $selected = '') // Common function used for both batches and filters.
 | |
| 	{
 | |
| 		$optdiz = array('batch' => LAN_BATCH_LABEL_PREFIX.' ', 'filter'=> LAN_FILTER_LABEL_PREFIX.' ');
 | |
| 		$obj = $this->getController();
 | |
| 		$table = $obj->getTableName();
 | |
| 		$text = '';
 | |
| 		$textsingle = '';
 | |
| 				
 | |
| 
 | |
| 		$searchFieldOpts = array();
 | |
| 
 | |
| 		$fieldList = $obj->getFields();
 | |
| 		$fieldSort = ($type ==='batch') ? $obj->getBatchSort() : $obj->getFilterSort();
 | |
| 
 | |
| 		if(!empty($fieldSort))
 | |
| 		{
 | |
| 			e107::getMessage()->addDebug("Filtering by custom sort field: ".print_r($fieldSort,true));
 | |
| 			$fieldList = $this->sortFieldsByPriority($fieldList, $fieldSort);
 | |
| 		}
 | |
| 
 | |
| 		foreach($fieldList as $key=>$val)
 | |
| 		{
 | |
| 			if(!empty($val['search']))
 | |
| 			{
 | |
| 				$searchFieldOpts['searchfield__' .$key] = $val['title'];
 | |
| 			}
 | |
| 
 | |
| 			if(empty($val[$type])) // ie. filter = false or batch = false.
 | |
| 			{
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			$option = array();
 | |
| 			$parms = vartrue($val['writeParms'], array());
 | |
| 			if(is_string($parms))
 | |
| 			{
 | |
| 				parse_str($parms, $parms);
 | |
| 			}
 | |
| 
 | |
| 			//Basic batch support for dropdown with multiple values. (comma separated)
 | |
| 			if(!empty($val['writeParms']['multiple']) && $val['type'] === 'dropdown' && !empty($val['writeParms']['optArray']))
 | |
| 			{
 | |
| 				$val['type'] = 'comma';
 | |
| 				$parms = $val['writeParms']['optArray'];
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 
 | |
| 			switch($val['type'])
 | |
| 			{
 | |
| 
 | |
| 					case 'text';
 | |
| 
 | |
| 						if(!empty($parms['sef']))
 | |
| 						{
 | |
| 							$option['sefgen__'.$key.'__'.$parms['sef']] = LAN_GENERATE;
 | |
| 						}
 | |
| 
 | |
| 						$searchFieldOpts['searchfield__' .$key] = $val['title'];
 | |
| 
 | |
| 					break;
 | |
| 
 | |
| 
 | |
| 					case 'number';
 | |
| 						if($type === 'filter')
 | |
| 						{
 | |
| 							$option[$key.'___ISEMPTY_'] = LAN_UI_FILTER_IS_EMPTY;
 | |
| 						}
 | |
| 
 | |
| 						$searchFieldOpts['searchfield__' .$key] = $val['title'];
 | |
| 
 | |
| 					break;
 | |
| 
 | |
| 					case 'textarea':
 | |
| 					case 'tags':
 | |
| 						$searchFieldOpts['searchfield__' .$key] = $val['title'];
 | |
| 					break;
 | |
| 
 | |
| 					case 'bool':
 | |
| 					case 'boolean': //TODO modify description based on $val['parm]
 | |
| 
 | |
| 						// defaults
 | |
| 						$LAN_TRUE = LAN_ON;
 | |
| 						$LAN_FALSE = LAN_OFF;
 | |
| 
 | |
| 						if(varset($parms['label']) === 'yesno')
 | |
| 						{
 | |
| 							$LAN_TRUE = LAN_YES;
 | |
| 							$LAN_FALSE = LAN_NO;
 | |
| 						}
 | |
| 						
 | |
| 						if(!empty($parms['enabled']))
 | |
| 						{
 | |
| 							$LAN_TRUE = $parms['enabled'];
 | |
| 						}
 | |
| 						elseif(!empty($parms['true']))
 | |
| 						{
 | |
| 							$LAN_TRUE = $parms['true'];
 | |
| 						}
 | |
| 
 | |
| 						if(!empty($parms['disabled']))
 | |
| 						{
 | |
| 							$LAN_FALSE = $parms['disabled'];
 | |
| 						}
 | |
| 						elseif(!empty($parms['false']))
 | |
| 						{
 | |
| 							$LAN_FALSE = $parms['false'];
 | |
| 						}
 | |
| 
 | |
| 						if(!empty($parms['reverse'])) // reverse true/false values;
 | |
| 						{
 | |
| 							$option['bool__'.$key.'__0'] = $LAN_TRUE;	// see newspost.php : news_allow_comments for an example.
 | |
| 							$option['bool__'.$key.'__1'] = $LAN_FALSE;
 | |
| 						}
 | |
| 						else 
 | |
| 						{
 | |
| 							$option['bool__'.$key.'__1'] = $LAN_TRUE;
 | |
| 							$option['bool__'.$key.'__0'] = $LAN_FALSE;
 | |
| 						}
 | |
| 							
 | |
| 						if($type === 'batch')
 | |
| 						{
 | |
| 							$option['boolreverse__'.$key] = LAN_BOOL_REVERSE;
 | |
| 						}
 | |
| 					break;
 | |
| 					
 | |
| 					case 'checkboxes':
 | |
| 					case 'comma':
 | |
| 
 | |
| 						if (!empty($parms['optArray']))
 | |
| 						{
 | |
| 							$fopts = $parms;
 | |
| 							$parms = $fopts['optArray'];
 | |
| 							unset($fopts['optArray']);
 | |
| 							$parms['__options'] = $fopts;
 | |
| 						}
 | |
| 				
 | |
| 						// TODO lan
 | |
| 						if(!isset($parms['__options']))
 | |
| 						{
 | |
| 							$parms['__options'] = array();
 | |
| 						}
 | |
| 						if(!is_array($parms['__options']))
 | |
| 						{
 | |
| 							parse_str($parms['__options'], $parms['__options']);
 | |
| 						}
 | |
| 						$opts = $parms['__options'];
 | |
| 						unset($parms['__options']); //remove element options if any
 | |
| 						
 | |
| 						$options = $parms ? $parms : array();
 | |
| 						if(empty($options))
 | |
| 						{
 | |
| 							continue 2;
 | |
| 						}
 | |
| 						
 | |
| 						
 | |
| 						if($type === 'batch')
 | |
| 						{
 | |
| 							$_option = array(); 
 | |
| 
 | |
| 							if(isset($options['addAll']))
 | |
| 							{
 | |
| 								$option['attach_all__'.$key] = vartrue($options['addAll'], '(' .LAN_ADD_ALL. ')');
 | |
| 								unset($options['addAll']);
 | |
| 							}
 | |
| 							if(isset($options['clearAll']))
 | |
| 							{
 | |
| 								$_option['deattach_all__'.$key] = vartrue($options['clearAll'], '(' .LAN_CLEAR_ALL. ')');
 | |
| 								unset($options['clearAll']);
 | |
| 							}
 | |
| 							
 | |
| 							if(!empty($opts['simple']))
 | |
| 							{
 | |
| 								foreach ($options as $value) 
 | |
| 								{
 | |
| 									$option['attach__'.$key.'__'.$value] = LAN_ADD. ' ' .$value;
 | |
| 									$_option['deattach__'.$key.'__'.$value] = LAN_REMOVE. ' ' .$value;
 | |
| 								}
 | |
| 							}
 | |
| 							else 
 | |
| 							{
 | |
| 								foreach ($options as $value => $label) 
 | |
| 								{
 | |
| 									$option['attach__'.$key.'__'.$value] = LAN_ADD. ' ' .$label;
 | |
| 									$_option['deattach__'.$key.'__'.$value] = LAN_REMOVE. ' ' .$label;
 | |
| 								}
 | |
| 							}
 | |
| 							$option = array_merge($option, $_option);
 | |
| 							unset($_option);
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							unset($options['addAll'], $options['clearAll']);
 | |
| 							if(!empty($opts['simple']))
 | |
| 							{
 | |
| 								foreach($options as $k)
 | |
| 								{
 | |
| 									$option[$key.'__'.$k] = $k;
 | |
| 								}
 | |
| 							}
 | |
| 							else
 | |
| 							{
 | |
| 								foreach($options as $k => $name)
 | |
| 								{
 | |
| 									$option[$key.'__'.$k] = $name;
 | |
| 								}
 | |
| 							}
 | |
| 
 | |
| 						}						
 | |
| 					break;
 | |
| 						
 | |
| 					case 'templates':
 | |
| 					case 'layouts':
 | |
| 						$parms['raw'] = true;
 | |
| 						$val['writeParms'] = $parms;
 | |
| 						$tmp = $this->renderElement($key, '', $val);
 | |
| 						if(is_array($tmp))
 | |
| 						{	
 | |
| 							foreach ($tmp as $k => $name)
 | |
| 							{
 | |
| 								$option[$key.'__'.$k] = $name;
 | |
| 							}
 | |
| 						}
 | |
| 					break;
 | |
| 
 | |
| 					case 'dropdown': // use the array $parm;
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 						if(!empty($parms['optArray']))
 | |
| 						{
 | |
| 							$fopts = $parms;
 | |
| 							$parms = $fopts['optArray'];
 | |
| 							unset($fopts['optArray']);
 | |
| 							$parms['__options'] = $fopts;
 | |
| 						}
 | |
| 
 | |
| 						if (!isset($parms['__options'])) $parms['__options'] = null;
 | |
| 						if(!is_array($parms['__options']))
 | |
| 						{
 | |
| 							parse_str((string) $parms['__options'], $parms['__options']);
 | |
| 						}
 | |
| 						$opts = $parms['__options'];
 | |
| 						if(!empty($opts['multiple']) && $type === 'batch')
 | |
| 						{
 | |
| 							// no batch support for multiple, should have some for filters soon
 | |
| 							continue 2;
 | |
| 						}
 | |
| 
 | |
| 						unset($parms['__options']); //remove element options if any
 | |
| 
 | |
| 
 | |
| 
 | |
| 						foreach($parms as $k => $name)
 | |
| 						{
 | |
| 							$option[$key.'__'.$k] = $name;
 | |
| 						}
 | |
| 					break;
 | |
| 
 | |
| 					case 'language': // full list of 
 | |
| 					case 'lanlist': // use the array $parm;
 | |
| 						if(!is_array(varset($parms['__options'])))
 | |
| 						{
 | |
| 							parse_str($parms['__options'], $parms['__options']);
 | |
| 						}
 | |
| 						$opts = $parms['__options'];
 | |
| 						if(!empty($opts['multiple']))
 | |
| 						{
 | |
| 							// no batch support for multiple, should have some for filters soon
 | |
| 							continue 2;
 | |
| 						}
 | |
| 						$options = ($val['type'] === 'language') ? e107::getLanguage()->getList() : e107::getLanguage()->getLanSelectArray();
 | |
| 						foreach($options as $code => $name)
 | |
| 						{
 | |
| 							$option[$key.'__'.$code] = $name;
 | |
| 						}
 | |
| 					break;
 | |
| 
 | |
| 					case 'datestamp':
 | |
| 						$tp = e107::getParser();
 | |
| 						if($type !== 'batch')
 | |
| 						{
 | |
| 							$dateFilters = array (
 | |
| 								'hour'   => LAN_UI_FILTER_PAST_HOUR,
 | |
| 								'day'    => LAN_UI_FILTER_PAST_24_HOURS,
 | |
| 								'week'   => LAN_UI_FILTER_PAST_WEEK,
 | |
| 								'month'  => LAN_UI_FILTER_PAST_MONTH,
 | |
| 								'month3' => $tp->lanVars(LAN_UI_FILTER_PAST_XMONTHS,3),
 | |
| 								'month6' => $tp->lanVars(LAN_UI_FILTER_PAST_XMONTHS,6),
 | |
| 								'month9' => $tp->lanVars(LAN_UI_FILTER_PAST_XMONTHS,9),
 | |
| 								'year'   => LAN_UI_FILTER_PAST_YEAR,
 | |
| 								'today' => LAN_UI_FILTER_TODAY,
 | |
| 								'thisweek' => LAN_UI_FILTER_THIS_WEEK,
 | |
| 								'thismonth' => LAN_UI_FILTER_THIS_MONTH,
 | |
| 								'thisyear' => LAN_UI_FILTER_THIS_YEAR,
 | |
| 							);
 | |
| 
 | |
| 							$dateFiltersFuture = array (
 | |
| 								'nhour'   => LAN_UI_FILTER_NEXT_HOUR,
 | |
| 								'nday'    => LAN_UI_FILTER_NEXT_24_HOURS,
 | |
| 								'nweek'   => LAN_UI_FILTER_NEXT_WEEK,
 | |
| 								'nmonth'  => LAN_UI_FILTER_NEXT_MONTH,
 | |
| 								'nmonth3' => $tp->lanVars(LAN_UI_FILTER_NEXT_XMONTHS,3),
 | |
| 								'nmonth6' => $tp->lanVars(LAN_UI_FILTER_NEXT_XMONTHS,6),
 | |
| 								'nmonth9' => $tp->lanVars(LAN_UI_FILTER_NEXT_XMONTHS,9),
 | |
| 								'nyear'   => LAN_UI_FILTER_NEXT_YEAR
 | |
| 							);
 | |
| 
 | |
| 							if($val['filter'] === 'future' )
 | |
| 							{
 | |
| 								$dateFilters = $dateFiltersFuture;
 | |
| 							}
 | |
| 
 | |
| 							if($val['filter'] === 'both')
 | |
| 							{
 | |
| 								$dateFilters += $dateFiltersFuture;
 | |
| 							}
 | |
| 
 | |
| 							foreach($dateFilters as $k => $name)
 | |
| 							{
 | |
| 								$option['datestamp__'.$key.'__'.$k] = $name;
 | |
| 							}
 | |
| 						}
 | |
| 						else // batch
 | |
| 						{
 | |
| 							$time = time();
 | |
| 							$option[$key.'__'.$time] = LAN_UI_BATCH_NOW;
 | |
| 						}
 | |
| 					    
 | |
| 
 | |
| 					break;
 | |
| 
 | |
| 					case 'userclass':
 | |
| 						$classes = e107::getUserClass()->uc_required_class_list(vartrue($parms['classlist'], 'public,nobody,guest,member,admin,main,classes'));
 | |
| 						foreach($classes as $k => $name)
 | |
| 						{
 | |
| 							$option[$key.'__'.$k] = $name;
 | |
| 						}
 | |
| 					break;
 | |
| 					case 'userclasses':
 | |
| 						$classes = e107::getUserClass()->uc_required_class_list(vartrue($parms['classlist'], 'public,nobody,guest,member,admin,main,classes'));
 | |
| 						$_option = array();
 | |
| 						
 | |
| 						if($type === 'batch')
 | |
| 						{
 | |
| 							foreach ($classes as $k => $v) 
 | |
| 							{
 | |
| 								$option['ucadd__'.$key.'__'.$k] = LAN_ADD.' '.$v;
 | |
| 								$_option['ucremove__'.$key.'__'.$k] = LAN_REMOVE. ' ' .$v;
 | |
| 							}
 | |
| 							$option['ucaddall__'.$key] = '(' .LAN_ADD_ALL. ')';
 | |
| 							$_option['ucdelall__'.$key] = '(' .LAN_CLEAR_ALL. ')';
 | |
| 							$option = array_merge($option, $_option);
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							foreach ($classes as $k => $v) 
 | |
| 							{
 | |
| 								$option[$key.'__'.$k] = $v;	
 | |
| 							}
 | |
| 						}
 | |
| 						
 | |
| 						unset($_option);
 | |
| 					break;
 | |
| 
 | |
| 					case 'method':
 | |
| 
 | |
| 						$method = !empty($val['method']) ? $val['method'] : $key; // Use the method attribute if specified, otherwise fall back to the field key
 | |
| 
 | |
| 						if(strpos($method, '::') !== false)
 | |
| 						{
 | |
| 							list($className, $methodName) = explode('::', $method);
 | |
| 							$cls = new $className();
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							$cls = $this;
 | |
| 							$methodName = $method;
 | |
| 						}
 | |
| 
 | |
| 						if(method_exists($cls, $methodName))
 | |
| 						{
 | |
| 							$list = call_user_func_array(array($cls, $methodName), array('', $type, $parms));
 | |
| 							if(is_array($list))
 | |
| 							{
 | |
| 								// Check for single option
 | |
| 								if(isset($list['singleOption']))
 | |
| 								{
 | |
| 									$textsingle .= $list['singleOption'];
 | |
| 									continue 2;
 | |
| 								}
 | |
| 								// options array
 | |
| 								foreach($list as $k => $name)
 | |
| 								{
 | |
| 									$option[$key . '__' . $k] = $name;
 | |
| 								}
 | |
| 							}
 | |
| 							elseif(!empty($list))
 | |
| 							{
 | |
| 								$text .= $list;
 | |
| 								continue 2;
 | |
| 							}
 | |
| 						}
 | |
| 						else
 | |
| 						{
 | |
| 							e107::getDebug()->log('Missing Method: ' . get_class($cls) . '::' . $methodName);
 | |
| 						}
 | |
| 						break;
 | |
| 
 | |
| 					case 'user': // TODO - User Filter
 | |
| 					
 | |
| 						$sql = e107::getDb();
 | |
| 						$field = $val['field'];
 | |
| 						
 | |
| 						$query = 'SELECT d.' .$field. ', u.user_name FROM #' .$val['table']. ' AS d LEFT JOIN #user AS u ON d.' .$field. ' = u.user_id  GROUP BY d.' .$field. ' ORDER BY u.user_name';
 | |
| 						$row = $sql->retrieve($query,true);
 | |
| 						foreach($row as $data)
 | |
| 						{
 | |
| 							$k = $data[$field];
 | |
| 							if($k == 0)
 | |
| 							{
 | |
| 								$option[$key.'__'.$k] = '(' .LAN_ANONYMOUS. ')';
 | |
| 							}
 | |
| 							else 
 | |
| 							{
 | |
| 								$option[$key.'__'.$k] = vartrue($data['user_name'],LAN_UNKNOWN);
 | |
| 							}
 | |
| 							
 | |
| 							
 | |
| 						}
 | |
| 					break;
 | |
| 			}
 | |
| 
 | |
| 
 | |
| 
 | |
| 			if(!empty($option))
 | |
| 			{
 | |
| 				$text .= "\t".$this->optgroup_open($optdiz[$type].defset($val['title'], $val['title']), varset($disabled))."\n";
 | |
| 
 | |
| 				foreach($option as $okey=>$oval)
 | |
| 				{
 | |
| 					$text .= $this->option($oval, $okey, $selected == $okey)."\n";
 | |
| 				}
 | |
| 				$text .= "\t".$this->optgroup_close()."\n";
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$text2 = '';
 | |
| 
 | |
| 		if(!empty($searchFieldOpts) && $type !== 'batch')
 | |
| 		{
 | |
| 			$text2 .= "\t".$this->optgroup_open(defset('LAN_UI_FILTER_SEARCH_IN_FIELD', 'Search in Field'))."\n";
 | |
| 
 | |
| 			foreach($searchFieldOpts as $key=>$val)
 | |
| 			{
 | |
| 				$text2 .= $this->option($val, $key, $selected == $key)."\n";
 | |
| 			}
 | |
| 
 | |
| 			$text2 .= "\t".$this->optgroup_close()."\n";
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 		return $textsingle.$text2.$text;
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getElementId()
 | |
| 	{
 | |
| 		$controller = $this->getController();
 | |
| 		$name = str_replace('_', '-', ($controller->getPluginName() === 'core' ? 'core-'.$controller->getTableName() : 'plugin-'.$controller->getPluginName()));
 | |
| 		return e107::getForm()->name2id($name); // prevent invalid ids.
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @return e_admin_ui
 | |
| 	 */
 | |
| 	public function getController()
 | |
| 	{
 | |
| 
 | |
| 		return $this->_controller;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| e107::loadAdminIcons();
 | |
| 
 | |
| /**
 | |
|  * TODO:
 | |
|  * 1. [DONE - a good start] move abstract peaces of code to the proper classes
 | |
|  * 2. [DONE - at least for alpha release] remove duplicated code (e_form & e_admin_form_ui), refactoring
 | |
|  * 3. make JS Manager handle Styles (.css files and inline CSS)
 | |
|  * 4. [DONE] e_form is missing some methods used in e_admin_form_ui
 | |
|  * 5. [DONE] date convert needs string-to-datestamp auto parsing, strptime() is the solution but needs support for
 | |
|  * 		Windows and PHP < 5.1.0 - build custom strptime() function (php_compatibility_handler.php) on this -
 | |
|  * 		http://sauron.lionel.free.fr/?page=php_lib_strptime (bad license so no copy/paste is allowed!)
 | |
|  * 6. [DONE - read/writeParms introduced ] $fields[parms] mess - fix it, separate list/edit mode parms somehow
 | |
|  * 7. clean up/document all object vars (e_admin_ui, e_admin_dispatcher)
 | |
|  * 8. [DONE hopefully] clean up/document all parameters (get/setParm()) in controller and model classes
 | |
|  * 9. [DONE] 'ip' field type - convert to human readable format while showing/editing record
 | |
|  * 10. draggable (or not?) ordering (list view)
 | |
|  * 11. [DONE] realtime search filter (typing text) - like downloads currently
 | |
|  * 12. [DONE] autosubmit when 'filter' dropdown is changed (quick fix?)
 | |
|  * 13. tablerender captions
 | |
|  * 14. [DONE] textareas auto-height
 | |
|  * 15. [DONE] multi JOIN table support (optional), aliases
 | |
|  * 16. tabs support (create/edit view)
 | |
|  * 17. tree list view (should handle cases like Site Links admin page)
 | |
|  */
 | |
|  
 | |
| 
 |