app->setOpenApp(); if(defined('IN_USE') or (defined('RUN_MODE') and RUN_MODE != 'api')) $this->setPreference(); if(!isset($this->config->bizVersion)) return false; /* Code for task #9224. Set requiredFields for workflow. */ if($this->dbh and (defined('IN_USE') or (defined('RUN_MODE') and RUN_MODE == 'api'))) { $this->checkRequireFlowField(); if(isset($this->config->{$this->moduleName}) and strpos($this->methodName, 'export') !== false) { if(isset($this->config->{$this->moduleName}->exportFields) or isset($this->config->{$this->moduleName}->list->exportFields)) { $exportFields = $this->dao->select('*')->from(TABLE_WORKFLOWFIELD)->where('module')->eq($this->moduleName)->andWhere('canExport')->eq('1')->andWhere('buildin')->eq('0')->fetchAll('field'); if(isset($this->config->{$this->moduleName}->exportFields)) { foreach($exportFields as $field) $this->config->{$this->moduleName}->exportFields .= ",{$field->field}"; } if(isset($this->config->{$this->moduleName}->list->exportFields)) { foreach($exportFields as $field) $this->config->{$this->moduleName}->list->exportFields .= ",{$field->field}"; } foreach($exportFields as $flowField => $exportField) { if(!isset($this->lang->{$this->moduleName}->$flowField)) $this->lang->{$this->moduleName}->$flowField = $exportField->name; } } } /* Append editor field to this module config from workflow. */ $textareaFields = $this->dao->select('*')->from(TABLE_WORKFLOWFIELD)->where('module')->eq($this->moduleName)->andWhere('control')->eq('richtext')->andWhere('buildin')->eq('0')->fetchAll('field'); if($textareaFields) { $editorIdList = array(); foreach($textareaFields as $textareaField) $editorIdList[] = $textareaField->field; if(!isset($this->config->{$this->moduleName})) $this->config->{$this->moduleName} = new stdclass(); if(!isset($this->config->{$this->moduleName}->editor)) $this->config->{$this->moduleName}->editor = new stdclass(); if(!isset($this->config->{$this->moduleName}->editor->{$this->methodName})) $this->config->{$this->moduleName}->editor->{$this->methodName} = array('id' => '', 'tools' => 'simpleTools'); $this->config->{$this->moduleName}->editor->{$this->methodName}['id'] .= ',' . join(',', $editorIdList); trim($this->config->{$this->moduleName}->editor->{$this->methodName}['id'], ','); } /* If workflow is created by a normal user, set priv. */ if(isset($this->app->user) and !$this->app->user->admin) { $actions = $this->dao->select('module, action')->from(TABLE_WORKFLOWACTION)->where('createdBy')->eq($this->app->user->account)->andWhere('buildin')->eq('0')->fetchGroup('module'); $labels = $this->dao->select('module, code')->from(TABLE_WORKFLOWLABEL)->where('createdBy')->eq($this->app->user->account)->andWhere('buildin')->eq('0')->fetchGroup('module'); if(!empty($actions)) { foreach($actions as $module => $actionObj) { foreach($actionObj as $action) $this->app->user->rights['rights'][$module][$action->action] = 1; } } if(!empty($labels)) { foreach($labels as $module => $codeObj) { foreach($codeObj as $code) { $code = str_replace('browse', '', $code->code); $this->app->user->rights['rights'][$module][$code] = 1; } } } } } } /** * Go to preference setting page if preference not setted. * * @access public * @return void */ public function setPreference() { if(empty($this->app->user->account)) return true; if($this->app->getModuleName() == 'user' and strpos("login,logout", $this->app->getMethodName()) !== false) return true; if($this->app->getModuleName() == 'my' and $this->app->getMethodName() == 'changepassword') return true; if($this->app->getModuleName() == 'my' and $this->app->getMethodName() == 'preference') return true; if(!isset($this->config->preferenceSetted) and $this->config->vision == 'rnd') { setcookie('tab', 'my', 0, $this->config->webRoot, '', $this->config->cookieSecure, false); $this->locate(helper::createLink('my', 'preference')); } } /** * 企业版部分功能是从然之合并过来的。然之代码中调用loadModel方法时传递了一个非空的appName,在禅道中会导致错误。 * 调用父类的loadModel方法来避免这个错误。 * Some codes merged from ranzhi called the function loadModel with a non-empty appName which causes an error in zentao. * Call the parent function with empty appName to avoid this error. * * @param string $moduleName 模块名,如果为空,使用当前模块。The module name, if empty, use current module's name. * @param string $appName The app name, if empty, use current app's name. * @access public * @return object|bool 如果没有model文件,返回false,否则返回model对象。If no model file, return false, else return the model object. */ public function loadModel($moduleName = '', $appName = '') { return parent::loadModel($moduleName); } /** * 设置视图文件:主视图文件,扩展视图文件, 站点扩展视图文件,以及钩子脚本。 * Set view files: the main file, extension view file, site extension view file and hook files. * * @param string $moduleName module name * @param string $methodName method name * @access public * @return string the view file */ public function setViewFile($moduleName, $methodName) { $moduleName = strtolower(trim($moduleName)); $methodName = strtolower(trim($methodName)); $modulePath = $this->app->getModulePath($this->appName, $moduleName); $viewExtPath = $this->app->getModuleExtPath($this->appName, $moduleName, 'view'); $viewType = ($this->viewType == 'mhtml' or $this->viewType == 'xhtml') ? 'html' : $this->viewType; $mainViewFile = $modulePath . 'view' . DS . $this->devicePrefix . $methodName . '.' . $viewType . '.php'; /* If the main view file doesn't exist, set the device prefix to empty and reset the main view file. */ if(!file_exists($mainViewFile) and $this->app->clientDevice != 'mobile') { $originalPrefix = $this->devicePrefix; $this->devicePrefix = ''; $mainViewFile = $modulePath . 'view' . DS . $this->devicePrefix . $methodName . '.' . $viewType . '.php'; $this->devicePrefix = $originalPrefix; } $viewFile = $mainViewFile; if(!empty($viewExtPath)) { $commonExtViewFile = $viewExtPath['common'] . $this->devicePrefix . $methodName . ".{$viewType}.php"; $xuanExtViewFile = $viewExtPath['xuan'] . $this->devicePrefix . $methodName . ".{$viewType}.php"; $visionExtViewFile = $viewExtPath['vision'] . $this->devicePrefix . $methodName . ".{$viewType}.php"; $customExtViewFile = $viewExtPath['custom'] . $this->devicePrefix . $methodName . ".{$viewType}.php"; $siteExtViewFile = empty($viewExtPath['site']) ? '' : $viewExtPath['site'] . $this->devicePrefix . $methodName . ".{$viewType}.php"; /* Get ext files, site > custom > vision > common. */ if(!empty($siteExtViewFile) and file_exists($siteExtViewFile)) { $viewFile = $siteExtViewFile; } else if(file_exists($customExtViewFile)) { $viewFile = $customExtViewFile; } else if(!empty($viewExtPath['vision']) and file_exists($visionExtViewFile)) { $viewFile = $visionExtViewFile; } else if(file_exists($xuanExtViewFile)) { $viewFile = $xuanExtViewFile; } else if(file_exists($commonExtViewFile)) { $viewFile = $commonExtViewFile; } if(!is_file($viewFile)) die(js::error($this->lang->notPage) . js::locate('back')); /* Get ext hook files. */ if(empty($viewExtPath['vision'])) { $commonExtHookFiles = glob($viewExtPath['common'] . $this->devicePrefix . $methodName . ".*.{$viewType}.hook.php"); } else { $commonExtHookFiles = glob($viewExtPath['vision'] . $this->devicePrefix . $methodName . ".*.{$viewType}.hook.php"); } $xuanExtHookFiles = glob($viewExtPath['xuan'] . $this->devicePrefix . $methodName . ".*.{$viewType}.hook.php"); $customExtHookFiles = glob($viewExtPath['custom'] . $this->devicePrefix . $methodName . ".*.{$viewType}.hook.php"); $siteExtHookFiles = empty($viewExtPath['site']) ? '' : glob($viewExtPath['site'] . $this->devicePrefix . $methodName . ".*.{$viewType}.hook.php"); $extHookFiles = array_merge((array)$commonExtHookFiles, (array)$xuanExtHookFiles, (array)$customExtHookFiles, (array)$siteExtHookFiles); } if(!empty($extHookFiles)) return array('viewFile' => $viewFile, 'hookFiles' => $extHookFiles); return $viewFile; } /** * 默认渲染方法,适用于viewType = html的时候。 * Default parse method when viewType != json, like html. * * @param string $moduleName module name * @param string $methodName method name * @access public * @return void */ public function parseDefault($moduleName, $methodName) { /** * 设置视图文件。(PHP7有一个bug,不能直接$viewFile = $this->setViewFile())。 * Set viewFile. (Can't assign $viewFile = $this->setViewFile() directly because one php7's bug.) */ $results = $this->setViewFile($moduleName, $methodName); $viewFile = $results; if(is_array($results)) extract($results); /** * 获得当前页面的CSS和JS。 * Get css and js codes for current method. */ $css = $this->getCSS($moduleName, $methodName); $js = $this->getJS($moduleName, $methodName); /* If the js or css file doesn't exist, set the device prefix to empty and reset the js or css file. */ if($this->viewType == 'xhtml') { $originalPrefix = $this->devicePrefix; $this->devicePrefix = ''; $css .= $this->getCSS($moduleName, $methodName); $js .= $this->getJS($moduleName, $methodName); $this->devicePrefix = $originalPrefix; } if($css) $this->view->pageCSS = $css; if($js) $this->view->pageJS = $js; /** * 切换到视图文件所在的目录,以保证视图文件里面的include语句能够正常运行。 * Change the dir to the view file to keep the relative paths work. */ $currentPWD = getcwd(); chdir(dirname($viewFile)); /** * 使用extract和ob方法渲染$viewFile里面的代码。 * Use extract and ob functions to eval the codes in $viewFile. */ extract((array)$this->view); ob_start(); include $viewFile; if(isset($hookFiles)) foreach($hookFiles as $hookFile) if(file_exists($hookFile)) include $hookFile; $this->output .= ob_get_contents(); ob_end_clean(); /** * 渲染完毕后,再切换回之前的路径。 * At the end, chang the dir to the previous. */ chdir($currentPWD); } /** * Execute hooks of a method. * * @param int $objectID The id of an object. The object maybe a bug | build | feedback | product | productplan | project | release | story | task | testcase | testsuite | testtask. * @access public * @return void */ public function executeHooks($objectID) { if(!isset($this->config->bizVersion)) return false; $moduleName = $this->moduleName; return $this->$moduleName->executeHooks($objectID); } /** * Build operate menu of a method. * * @param object $object product|project|productplan|release|build|story|task|bug|testtask|testcase|testsuite * @param string $displayOn view|browse * @access public * @return void */ public function buildOperateMenu($object, $displayOn = 'view') { if(!isset($this->config->bizVersion)) return false; $flow = $this->loadModel('workflow')->getByModule($this->moduleName); return $this->loadModel('flow')->buildOperateMenu($flow, $object, $displayOn); } /** * Print extend fields. * * @param object $object bug | build | feedback | product | productplan | project | release | story | task | testcase | testsuite | testtask * @param string $type The parent component which fileds displayed in. It should be table or div. * @param string $extras The extra params. * columns=2|3|5 Number of the columns merged to display the fields. The default is 1. * position=left|right The position which the fields displayed in a page. * inForm=0|1 The fields displayed in a form or not. The default is 1. * inCell=0|1 The fields displayed in a div with class cell or not. The default is 0. * @access public * @return void */ public function printExtendFields($object, $type, $extras = '') { if(!isset($this->config->bizVersion)) return false; echo $this->loadModel('flow')->printFields($this->moduleName, $this->methodName, $object, $type, $extras); } /** * Process status of an object according to its subStatus. * * @param string $module product | release | story | project | task | bug | testcase | testtask | feedback * @param object $record a record of above modules. * @access public * @return string */ public function processStatus($module, $record) { $moduleName = $this->moduleName; return $this->$moduleName->processStatus($module, $record); } /** * Check require with flow field when post data. * * @access public * @return void */ public function checkRequireFlowField() { if($this->config->edition == 'open') return false; if(empty($_POST)) return false; $action = $this->dao->select('*')->from(TABLE_WORKFLOWACTION)->where('module')->eq($this->moduleName)->andWhere('action')->eq($this->methodName)->fetch(); if(empty($action)) return false; if($action->extensionType == 'none' and $action->buildin == 1) return false; $flow = $this->dao->select('*')->from(TABLE_WORKFLOW)->where('module')->eq($this->moduleName)->fetch(); $fields = $this->loadModel('workflowaction')->getFields($this->moduleName, $this->methodName); $layouts = $this->loadModel('workflowlayout')->getFields($this->moduleName, $this->methodName); $rules = $this->dao->select('*')->from(TABLE_WORKFLOWRULE)->orderBy('id_desc')->fetchAll('id'); $requiredFields = ''; $mustPostFields = ''; $numberFields = ''; $message = array(); foreach($fields as $field) { if(!empty($field->buildin)) continue; if(empty($field->show)) continue; if(!isset($layouts[$field->field])) continue; $fieldRules = explode(',', trim($field->rules, ',')); $fieldRules = array_unique($fieldRules); foreach($fieldRules as $ruleID) { if(!isset($rules[$ruleID])) continue; if(!empty($_POST[$field->field]) and !is_string($_POST[$field->field])) continue; $rule = $rules[$ruleID]; if($rule->type == 'system' and $rule->rule == 'notempty') { $requiredFields .= ",{$field->field}"; if($field->control == 'radio' or $field->control == 'checkbox') $mustPostFields .= ",{$field->field}"; if(strpos($field->type, 'int') !== false and $field->control == 'select') $numberFields .= ",{$field->field}"; } elseif($rule->type == 'system' and isset($_POST[$field->field])) { $pass = true; if($rule->rule == 'unique') { if(!empty($_POST[$field->field])) { $sqlClass = new sql(); $sql = "SELECT COUNT(*) AS count FROM $flow->table WHERE `$field->field` = " . $sqlClass->quote(fixer::input('post')->get($field->field)); if(isset($_POST['id'])) $sql .= ' AND `id` != ' . (int)$_POST['id']; $row = $this->dbh->query($sql)->fetch(); if($row->count != 0) $pass = false; } } else { $checkFunc = 'check' . $rule->rule; if(validater::$checkFunc($_POST[$field->field]) === false) $pass = false; } if(!$pass) { $error = zget($this->lang->error, $rule->rule, ''); if($rule->rule == 'unique') $error = sprintf($error, $field->name, $_POST[$field->field]); if($error) $error = sprintf($error, $field->name); if(empty($error)) $error = sprintf($this->lang->error->reg, $field->name, $rule->rule); $message[$field->field][] = $error; } } elseif($rule->type == 'regex' and isset($_POST[$field->field])) { if(validater::checkREG($_POST[$field->field], $rule->rule) === false) $message[$field->field][] = sprintf($this->lang->error->reg, $field->name, $rule->rule); } } } if($requiredFields) { if(isset($this->config->{$this->moduleName}->{$this->methodName}->requiredFields)) $requiredFields .= ',' . $this->config->{$this->moduleName}->{$this->methodName}->requiredFields; foreach(explode(',', $requiredFields) as $requiredField) { if(empty($requiredField)) continue; if(!isset($fields[$requiredField])) continue; if(isset($_POST[$requiredField]) and $_POST[$requiredField] === '') { $message[$requiredField][] = sprintf($this->lang->error->notempty, $fields[$requiredField]->name); } elseif(strpos(",{$numberFields},", ",{$requiredField},") !== false and empty($_POST[$requiredField])) { $message[$requiredField][] = sprintf($this->lang->error->notempty, $fields[$requiredField]->name); } elseif(strpos(",{$mustPostFields},", ",{$requiredField},") !== false and !isset($_POST[$requiredField])) { $message[$requiredField][] = sprintf($this->lang->error->notempty, $fields[$requiredField]->name); } } } if($message) $this->send(array('result' => 'fail', 'message' => $message)); } /** * Print view file. * * @param string $viewFile * @access public * @return void */ public function printViewFile($viewFile) { if(!file_exists($viewFile)) return false; $currentPWD = getcwd(); chdir(dirname($viewFile)); extract((array)$this->view); ob_start(); include $viewFile; $output = ob_get_contents(); ob_end_clean(); chdir($currentPWD); return $output; } }