567 lines
20 KiB
PHP
567 lines
20 KiB
PHP
<?php
|
|
/**
|
|
* The model file of editor module of ZenTaoCMS.
|
|
*
|
|
* @copyright Copyright 2009-2015 青岛易软天创网络科技有限公司(QingDao Nature Easy Soft Network Technology Co,LTD, www.cnezsoft.com)
|
|
* @license ZPL (http://zpl.pub/page/zplv11.html)
|
|
* @author Yidong Wang <yidong@cnezsoft.com>
|
|
* @package editor
|
|
* @version $Id$
|
|
* @link http://www.zentao.net
|
|
*/
|
|
class editorModel extends model
|
|
{
|
|
private $pathFix = DIRECTORY_SEPARATOR;
|
|
|
|
/**
|
|
* Get all modules
|
|
*
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function getModules()
|
|
{
|
|
$modules = glob($this->app->getModuleRoot() . '*');
|
|
$moduleList = "<div class='list-group'>";
|
|
foreach($modules as $module)
|
|
{
|
|
$module = basename($module);
|
|
if($module == 'editor' or $module == 'help' or $module == 'setting') continue;
|
|
$moduleName = !empty($this->lang->editor->modules[$module]) ? $this->lang->editor->modules[$module] : $module;
|
|
$moduleList .= html::a(inlink('extend', "moduleDir=$module"), $moduleName, 'extendWin', "class='list-group-item'");
|
|
}
|
|
$moduleList .= '</div>';
|
|
return $moduleList;
|
|
}
|
|
|
|
/**
|
|
* Get module files, contain control's methods and model's method but except ext.
|
|
*
|
|
* @access public
|
|
* @return array
|
|
*/
|
|
public function getModuleFiles($moduleDir)
|
|
{
|
|
$moduleRoot = $this->app->getModuleRoot();
|
|
$moduleFullDir = $moduleRoot . $moduleDir;
|
|
$moduleFiles = scandir($moduleFullDir);
|
|
foreach($moduleFiles as $moduleFile)
|
|
{
|
|
if($moduleFile == '.' or $moduleFile == '..' or $moduleFile == '.svn') continue;
|
|
$moduleFullFile = $moduleFullDir . $this->pathFix . $moduleFile;
|
|
if($moduleFile == 'control.php' or $moduleFile == 'model.php')
|
|
{
|
|
$allModules[$moduleFullDir][$moduleFullFile] = $this->analysis($moduleFullFile);
|
|
}
|
|
elseif($moduleFile == 'ext')
|
|
{
|
|
$allModules[$moduleFullDir][$moduleFullFile] = $this->getExtensionFiles($moduleFullFile);
|
|
}
|
|
elseif(is_dir($moduleFullFile))
|
|
{
|
|
$ext = ($moduleFile == 'js' or $moduleFile == 'css') ? $moduleFile : 'php';
|
|
foreach(glob($moduleFullFile . $this->pathFix . "*.$ext") as $fileName) $allModules[$moduleFullDir][$moduleFullFile][$fileName] = basename($fileName);
|
|
}
|
|
else
|
|
{
|
|
$allModules[$moduleFullDir][$moduleFullFile] = $moduleFile;
|
|
}
|
|
}
|
|
$allModules = $this->sortModule($moduleFullDir, $allModules);
|
|
return $allModules;
|
|
}
|
|
|
|
/**
|
|
* Get extension files.
|
|
*
|
|
* @param int $extPath
|
|
* @access public
|
|
* @return void
|
|
*/
|
|
public function getExtensionFiles($extPath)
|
|
{
|
|
$extensionList = array();
|
|
$extensionDirs = scandir($extPath);
|
|
foreach($extensionDirs as $extensionDir)
|
|
{
|
|
if($extensionDir == '.' or $extensionDir == '..' or $extensionDir == '.svn') continue;
|
|
$extensionFullDir = $extPath . $this->pathFix . $extensionDir;
|
|
if(is_dir($extensionFullDir))
|
|
{
|
|
$extensionList[$extensionFullDir] = array();
|
|
/* extend of lang is more a grade of directroy. */
|
|
if($extensionDir == 'lang' or $extensionDir == 'js' or $extensionDir == 'css')
|
|
{
|
|
|
|
$extensionList[$extensionFullDir] = $this->getTwoGradeFiles($extensionFullDir);
|
|
continue;
|
|
}
|
|
$extensionFiles = scandir($extensionFullDir);
|
|
foreach($extensionFiles as $extensionFile)
|
|
{
|
|
if($extensionFile == '.' or $extensionFile == '..' or $extensionFile == '.svn') continue;
|
|
$extensionFullFile = $extensionFullDir . $this->pathFix . $extensionFile;
|
|
$extensionList[$extensionFullDir][$extensionFullFile] = $extensionFile;
|
|
}
|
|
}
|
|
}
|
|
$extensionList = $this->sortModule($extPath, $extensionList, true);
|
|
return $extensionList;
|
|
}
|
|
|
|
/**
|
|
* Sort module
|
|
*
|
|
* @param string $filePath
|
|
* @param string $moduleFiles
|
|
* @param bool $isExtension
|
|
* @access public
|
|
* @return array
|
|
*/
|
|
public function sortModule($filePath, $moduleFiles, $isExtension = false)
|
|
{
|
|
$sort = $isExtension ? 'extSort' : 'sort';
|
|
$sortModules = array();
|
|
if(!$isExtension) $moduleFiles = $moduleFiles[$filePath];
|
|
foreach($this->config->editor->$sort as $sort)
|
|
{
|
|
$sortKey = empty($sort) ? $filePath : $filePath . $this->pathFix . $sort;
|
|
if(array_key_exists($sortKey, $moduleFiles)) $sortModules[$sortKey] = $moduleFiles[$sortKey];
|
|
}
|
|
return $sortModules;
|
|
}
|
|
|
|
/**
|
|
* if a directory has two grage, this method will get files
|
|
*
|
|
* @param string $extensionFullDir
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function getTwoGradeFiles($extensionFullDir)
|
|
{
|
|
$fileList = array();
|
|
$langDirs = scandir($extensionFullDir);
|
|
foreach($langDirs as $langDir)
|
|
{
|
|
if($langDir == '.' or $langDir == '..' or $langDir == '.svn') continue;
|
|
$langFullDir = $extensionFullDir . $this->pathFix . $langDir;
|
|
$fileList[$langFullDir] = array();
|
|
if(is_dir($langFullDir))
|
|
{
|
|
$langFiles = scandir($langFullDir);
|
|
foreach($langFiles as $langFile)
|
|
{
|
|
if($langFile == '.' or $langFile == '..' or $langFile == '.svn') continue;
|
|
$langFullFile = $langFullDir . $this->pathFix . $langFile;
|
|
$fileList[$langFullDir][$langFullFile] = $langFile;
|
|
}
|
|
}
|
|
}
|
|
return $fileList;
|
|
}
|
|
|
|
/**
|
|
* Analysis methods of control and model.
|
|
*
|
|
* @param string $fileName
|
|
* @access public
|
|
* @return array
|
|
*/
|
|
public function analysis($fileName)
|
|
{
|
|
$classMethod = array();
|
|
$class = strstr($fileName, $this->pathFix . 'module' . $this->pathFix);
|
|
$class = substr($class, 0, strpos($class, $this->pathFix, 9));
|
|
$class = basename($class);
|
|
if(strpos($fileName, 'model.php') !== false) $class .= 'Model';
|
|
if(!class_exists($class)) include $fileName;
|
|
$reflection = new ReflectionClass($class);
|
|
foreach($reflection->getMethods(ReflectionMethod::IS_PUBLIC) as $method)
|
|
{
|
|
$methodName = $method->name;
|
|
if($method->getFileName() != $fileName) continue;
|
|
if($methodName == '__construct') continue;
|
|
$classMethod[$fileName . $this->pathFix . $methodName] = $methodName;
|
|
}
|
|
return $classMethod;
|
|
}
|
|
|
|
/**
|
|
* Print tree from module files.
|
|
*
|
|
* @param int $files
|
|
* @access public
|
|
* @return void
|
|
*/
|
|
public function printTree($files, $isRoot = true)
|
|
{
|
|
if(empty($files) or !is_array($files)) return false;
|
|
$tree = $isRoot ? "<ul id='extendTree' class='tree treeview'>\n" : "<ul>\n";
|
|
if($isRoot)
|
|
{
|
|
$module = basename(dirname(key($files)));
|
|
|
|
$langFile = dirname(key($files)) . $this->pathFix . 'lang' . $this->pathFix . $this->cookie->lang. '.php';
|
|
if(file_exists($langFile))
|
|
{
|
|
if(!isset($lang)) $lang = new stdclass();
|
|
if(!isset($lang->$module)) $lang->$module = new stdclass();
|
|
include_once $langFile;
|
|
}
|
|
|
|
if(isset($lang->$module))
|
|
{
|
|
$this->module = $lang->$module;
|
|
}
|
|
elseif(isset($this->lang->$module))
|
|
{
|
|
$this->module = $this->lang->$module;
|
|
}
|
|
else
|
|
{
|
|
$this->module = '';
|
|
}
|
|
}
|
|
foreach($files as $key => $file)
|
|
{
|
|
$tree .= "<li>\n";
|
|
if(is_array($file))
|
|
{
|
|
$tree .= $this->addLink4Dir($key);
|
|
$tree .= $this->printTree($file, false);
|
|
}
|
|
else
|
|
{
|
|
$tree .= $this->addLink4File($key, $file);
|
|
}
|
|
$tree .= "</li>\n";
|
|
}
|
|
$tree .= "</ul>\n";
|
|
return $tree;
|
|
}
|
|
|
|
/**
|
|
* Add link for directory or has children grade
|
|
*
|
|
* @param string $filePath
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function addLink4Dir($filePath)
|
|
{
|
|
$tree = '';
|
|
$fileName = basename($filePath);
|
|
if(isset($this->lang->editor->modules[$fileName]))
|
|
{
|
|
$file = "<span title='$fileName'>" . $this->lang->editor->modules[$fileName] . '</span>';
|
|
}
|
|
elseif(isset($this->lang->editor->translate[$fileName]))
|
|
{
|
|
$file = "<span title='$fileName'>" . $this->lang->editor->translate[$fileName] . '</span>';
|
|
}
|
|
else
|
|
{
|
|
$file = "<span title='$fileName'>$fileName</span>";
|
|
}
|
|
if(strpos($filePath, $this->pathFix . 'ext' . $this->pathFix) !== false)
|
|
{
|
|
switch($fileName)
|
|
{
|
|
case 'lang': $tree .= $file; break;
|
|
case 'js': $tree .= "$file " . html::a($this->getExtendLink($filePath, "newJS"), $this->lang->editor->newExtend, 'editWin'); break;
|
|
case 'css': $tree .= "$file " . html::a($this->getExtendLink($filePath, "newCSS"), $this->lang->editor->newExtend, 'editWin'); break;
|
|
default: $tree .= "$file " . html::a($this->getExtendLink($filePath, "newExtend"), $this->lang->editor->newExtend, 'editWin');
|
|
}
|
|
}
|
|
elseif($fileName == 'model.php')
|
|
{
|
|
$tree .= "$file " . html::a($this->getExtendLink($filePath, 'newMethod'), $this->lang->editor->newMethod, 'editWin');
|
|
}
|
|
elseif($fileName == 'control.php')
|
|
{
|
|
$tree .= "$file " . html::a(inlink('newPage', "filePath=" . helper::safe64Encode($filePath)), $this->lang->editor->newPage, 'editWin');
|
|
}
|
|
else
|
|
{
|
|
$tree .= $file;
|
|
}
|
|
return $tree;
|
|
}
|
|
|
|
/**
|
|
* Add link for file
|
|
*
|
|
* @param string $filePath
|
|
* @param string $file
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function addLink4File($filePath, $file)
|
|
{
|
|
$tree = '';
|
|
$file = "<span title='$file'>$file</span>";
|
|
if(strpos($filePath, $this->pathFix . 'ext' . $this->pathFix) !== false)
|
|
{
|
|
$tree .= "$file " . html::a($this->getExtendLink($filePath, "edit"), $this->lang->edit, 'editWin');
|
|
$tree .= html::a(inlink('delete', 'path=' . helper::safe64Encode($filePath)), $this->lang->delete, 'hiddenwin') . "\n";
|
|
}
|
|
elseif(basename(dirname($filePath))== 'view')
|
|
{
|
|
$tree .= "$file " . html::a($this->getExtendLink($filePath, "override"), $this->lang->editor->override, 'editWin');
|
|
$tree .= html::a($this->getExtendLink($filePath, "newHook"), $this->lang->editor->newHook, 'editWin') . "\n";
|
|
}
|
|
else
|
|
{
|
|
$parentDir = basename(dirname($filePath));
|
|
$action = 'extendOther';
|
|
if($parentDir == 'control.php') $action = 'extendControl';
|
|
if($parentDir == 'model.php') $action = 'extendModel';
|
|
|
|
$tree .= "$file " . html::a($this->getExtendLink($filePath, $action), $this->lang->editor->extend, 'editWin');
|
|
if($action != 'extendOther') $tree .= html::a($this->getAPILink($filePath, $action), $this->lang->editor->api, 'editWin');
|
|
if($parentDir == 'lang') $tree .= html::a($this->getExtendLink($filePath, "new" . str_replace('-', '_', basename($filePath, '.php'))), $this->lang->editor->newLang, 'editWin');
|
|
if(basename($filePath) == 'config.php') $tree .= html::a($this->getExtendLink($filePath, "newConfig"), $this->lang->editor->newConfig, 'editWin');
|
|
}
|
|
return $tree;
|
|
}
|
|
|
|
/**
|
|
* Get extend link
|
|
*
|
|
* @param string $filePath
|
|
* @param string $action
|
|
* @param string $isExtends
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function getExtendLink($filePath, $action, $isExtends = '')
|
|
{
|
|
return inlink('edit', "filePath=" . helper::safe64Encode($filePath) . "&action=$action&isExtends=$isExtends");
|
|
}
|
|
|
|
/**
|
|
* Get api link.
|
|
*
|
|
* @param int $filePath
|
|
* @param int $action
|
|
* @param string $type
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function getAPILink($filePath, $action)
|
|
{
|
|
return helper::createLink('api', 'debug', "filePath=" . helper::safe64Encode($filePath) . "&action=$action");
|
|
}
|
|
|
|
/**
|
|
* Save file to extension.
|
|
*
|
|
* @param string $filePath
|
|
* @access public
|
|
* @return void
|
|
*/
|
|
public function save($filePath)
|
|
{
|
|
$fileContent = $this->post->fileContent;
|
|
if(get_magic_quotes_gpc()) $fileContent = stripslashes($fileContent);
|
|
$dirPath = dirname($filePath);
|
|
$extFilePath = substr($filePath, 0, strpos($filePath, $this->pathFix . 'ext' . $this->pathFix) + 4);
|
|
if(!is_dir($dirPath) and is_writable($extFilePath)) mkdir($dirPath, 0777, true);
|
|
if(is_writable($dirPath))
|
|
{
|
|
file_put_contents($filePath, $fileContent);
|
|
}
|
|
else
|
|
{
|
|
die(js::alert($this->lang->editor->notWritable . $extFilePath));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Extend model.php and get file content.
|
|
*
|
|
* @param string $filePath
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function extendModel($filePath)
|
|
{
|
|
$className = basename(dirname(dirname($filePath)));
|
|
if(!class_exists($className)) include(dirname($filePath));
|
|
$methodName = basename($filePath);
|
|
$methodParam = $this->getParam($className, $methodName, 'Model');
|
|
return $fileContent = <<<EOD
|
|
<?php
|
|
public function $methodName($methodParam)
|
|
{
|
|
return parent::$methodName($methodParam);
|
|
}
|
|
EOD;
|
|
}
|
|
|
|
/**
|
|
* Extend control.php and get file content.
|
|
*
|
|
* @param string $filePath
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function extendControl($filePath, $isExtends)
|
|
{
|
|
$className = basename(dirname(dirname($filePath)));
|
|
if(!class_exists($className)) include(dirname($filePath));
|
|
$methodName = basename($filePath);
|
|
if($isExtends == 'yes')
|
|
{
|
|
$methodParam = $this->getParam($className, $methodName);
|
|
return $fileContent = <<<EOD
|
|
<?php
|
|
include '../../control.php';
|
|
class my$className extends $className
|
|
{
|
|
public function $methodName($methodParam)
|
|
{
|
|
return parent::$methodName($methodParam);
|
|
}
|
|
}
|
|
EOD;
|
|
}
|
|
else
|
|
{
|
|
$methodCode = $this->getMethodCode($className, $methodName);
|
|
return $fileContent = <<<EOD
|
|
<?php
|
|
class $className extends control
|
|
{
|
|
$methodCode
|
|
}
|
|
EOD;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add a control method.
|
|
*
|
|
* @param string $filePath
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function newControl($filePath)
|
|
{
|
|
$className = strstr($filePath, $this->pathFix . 'module' . $this->pathFix);
|
|
$className = substr($className, 0, strpos($className, $this->pathFix, 9));
|
|
$className = basename($className);
|
|
$methodName = basename($filePath, '.php');
|
|
return $fileContent = <<<EOD
|
|
<?php
|
|
class $className extends control
|
|
{
|
|
public function $methodName()
|
|
{
|
|
}
|
|
}
|
|
EOD;
|
|
}
|
|
|
|
/**
|
|
* Get method's parameters.
|
|
*
|
|
* @param string $className
|
|
* @param string $methodName
|
|
* @param string $ext
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function getParam($className, $methodName, $ext = '')
|
|
{
|
|
$method = new ReflectionMethod($className . $ext, $methodName);
|
|
$methodParam = '';
|
|
foreach ($method->getParameters() as $param)
|
|
{
|
|
$methodParam .= '$' . $param->getName();
|
|
if($param->isOptional())
|
|
{
|
|
$defaultParam = $param->getDefaultValue();
|
|
if(is_string($defaultParam)) $methodParam .= "='$defaultParam', ";
|
|
else $methodParam .= "=$defaultParam, ";
|
|
}
|
|
else
|
|
{
|
|
$methodParam .= ', ';
|
|
}
|
|
}
|
|
$methodParam = rtrim($methodParam, ', ');
|
|
return $methodParam;
|
|
}
|
|
|
|
/**
|
|
* Get method code.
|
|
*
|
|
* @param string $className
|
|
* @param string $methodName
|
|
* @param string $ext value may be Model
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function getMethodCode($className, $methodName, $ext = '')
|
|
{
|
|
$method = new ReflectionMethod($className . $ext, $methodName);
|
|
$fileName = $method->getFileName();
|
|
$startLine = $method->getStartLine();
|
|
$endLine = $method->getEndLine();
|
|
$file = file($fileName);
|
|
$code = '';
|
|
for($i = $startLine - 1; $i <= $endLine; $i++)
|
|
{
|
|
$code .= $file[$i];
|
|
}
|
|
return $code;
|
|
}
|
|
|
|
/**
|
|
* Get save path.
|
|
*
|
|
* @param string $filePath
|
|
* @param string $action
|
|
* @access public
|
|
* @return string
|
|
*/
|
|
public function getSavePath($filePath, $action)
|
|
{
|
|
$fileName = empty($_POST['fileName']) ? '' : trim($this->post->fileName);
|
|
$moduleName = strstr($filePath, $this->pathFix . 'module' . $this->pathFix);
|
|
$moduleName = substr($moduleName, 0, strpos($moduleName, $this->pathFix, 9));
|
|
$moduleName = basename($moduleName);
|
|
$extPath = $this->app->getModuleRoot() . $moduleName . $this->pathFix . 'ext' . $this->pathFix;
|
|
switch($action)
|
|
{
|
|
case 'extendModel':
|
|
$fileName = empty($fileName) ? strtolower(basename($filePath)) . '.php' : $fileName;
|
|
return $extPath . 'model' . $this->pathFix . $fileName;
|
|
case 'extendControl':
|
|
$fileName = strtolower(basename($filePath)) . '.php';
|
|
return $extPath . 'control' . $this->pathFix . $fileName;
|
|
case 'override':
|
|
$fileName = basename($filePath);
|
|
return $extPath . 'view' . $this->pathFix . $fileName;
|
|
case 'extendOther':
|
|
$editName = basename($filePath);
|
|
$fileName = empty($fileName) ? $editName: $fileName;
|
|
if($editName == 'config.php') return $extPath . 'config' .$this->pathFix . $fileName;
|
|
elseif(strpos($editName, '.php') !== false) return $extPath . 'lang' . $this->pathFix . str_replace('.php', '', $editName) . $this->pathFix . $fileName;
|
|
else return $extPath . substr($editName, strrpos($editName, '.') + 1) . $this->pathFix . substr($editName, 0, strrpos($editName, '.')) . $this->pathFix . $fileName;
|
|
default:
|
|
if(empty($fileName)) die(js::error($this->lang->editor->emptyFileName));
|
|
$action = strtolower(str_replace('new', '', $action));
|
|
if($action == 'hook') return $extPath . 'view' . $this->pathFix . $fileName;
|
|
elseif($action == 'method') return $extPath . basename($filePath, '.php') . $this->pathFix . $fileName;
|
|
elseif($action == 'extend') return $filePath . $this->pathFix . $fileName;
|
|
elseif($action == 'config') return $extPath . 'config' . $this->pathFix . $fileName;
|
|
elseif($action == 'js') return $extPath . 'js' . $this->pathFix . substr($fileName, 0, strrpos($fileName, '.')) . $this->pathFix . $fileName;
|
|
elseif($action == 'css') return $extPath . 'css' . $this->pathFix . substr($fileName, 0, strrpos($fileName, '.')) . $this->pathFix . $fileName;
|
|
else return $extPath . 'lang' . $this->pathFix . str_replace('_', '-', $action) . $this->pathFix . $fileName;
|
|
}
|
|
}
|
|
}
|