app->user->admin) return true; return (strpos(",{$this->app->user->view->projects},", ",{$projectID},") !== false); } /** * Get Multiple linked products for project. * * @param int $projectID * @access public * @return array */ public function getMultiLinkedProducts($projectID) { $linkedProducts = $this->dao->select('product')->from(TABLE_PROJECTPRODUCT)->where('project')->eq($projectID)->fetchPairs(); $multiLinkedProducts = $this->dao->select('t3.id,t3.name')->from(TABLE_PROJECTPRODUCT)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.project = t2.id') ->leftJoin(TABLE_PRODUCT)->alias('t3')->on('t1.product = t3.id') ->where('t1.product')->in($linkedProducts) ->andWhere('t1.project')->ne($projectID) ->andWhere('t2.type')->eq('project') ->andWhere('t2.deleted')->eq('0') ->andWhere('t3.deleted')->eq('0') ->fetchPairs('id', 'name'); return $multiLinkedProducts; } /** * Show accessDenied response. * * @access private * @return void */ public function accessDenied() { if(defined('TUTORIAL')) return true; echo(js::alert($this->lang->project->accessDenied)); $this->session->set('project', ''); return print(js::locate(helper::createLink('project', 'index'))); } /** * Judge an action is clickable or not. * * @param object $project * @param string $action * @access public * @return bool */ public static function isClickable($project, $action) { $action = strtolower($action); if(empty($project)) return true; if(!isset($project->type)) return true; if($action == 'start') return $project->status == 'wait' or $project->status == 'suspended'; if($action == 'finish') return $project->status == 'wait' or $project->status == 'doing'; if($action == 'close') return $project->status != 'closed'; if($action == 'suspend') return $project->status == 'wait' or $project->status == 'doing'; if($action == 'activate') return $project->status == 'done' or $project->status == 'closed'; if($action == 'whitelist') return $project->acl != 'open'; if($action == 'group') return $project->model != 'kanban'; return true; } /** * Get budget unit list. * * @access public * @return array */ public function getBudgetUnitList() { $budgetUnitList = array(); if($this->config->vision != 'lite') { foreach(explode(',', $this->config->project->unitList) as $unit) $budgetUnitList[$unit] = zget($this->lang->project->unitList, $unit, ''); } return $budgetUnitList; } /** * Save project state. * * @param int $projectID * @param array $projects * @access public * @return int */ public function saveState($projectID = 0, $projects = array()) { if(defined('TUTORIAL')) return $projectID; if($projectID == 0 and $this->cookie->lastProject) $projectID = $this->cookie->lastProject; if($projectID == 0 and (int)$this->session->project == 0) $projectID = key($projects); if($projectID == 0) $projectID = key($projects); $this->session->set('project', (int)$projectID, $this->app->tab); if(!isset($projects[$this->session->project])) { if($projectID and strpos(",{$this->app->user->view->projects},", ",{$this->session->project},") === false and !empty($projects)) { /* Redirect old project to new project. */ $newProjectID = $this->dao->select('project')->from(TABLE_PROJECT)->where('id')->eq($projectID)->fetch('project'); if($newProjectID and strpos(",{$this->app->user->view->projects},", ",{$newProjectID},") !== false) { $this->session->set('project', (int)$newProjectID, $this->app->tab); return $this->session->project; } $this->session->set('project', key($projects), $this->app->tab); $this->accessDenied(); } } return $this->session->project; } /* * Get project swapper. * * @param int $projectID * @param string $currentModule * @param string $currentMethod * @access public * @return string */ public function getSwitcher($projectID, $currentModule, $currentMethod) { if($currentModule == 'project' and $currentMethod == 'browse') return; $currentProjectName = $this->lang->project->common; if($projectID) { $currentProject = $this->getById($projectID); $currentProjectName = $currentProject->name; } if($this->app->viewType == 'mhtml' and $projectID) { $output = $this->lang->project->common . $this->lang->colon; $output .= "{$currentProjectName} "; return $output; } $dropMenuLink = helper::createLink('project', 'ajaxGetDropMenu', "objectID=$projectID&module=$currentModule&method=$currentMethod"); $output = "
"; return $output; } /** * Get a project by id. * * @param int $projectID * @param string $type project|sprint,stage * @access public * @return object */ public function getByID($projectID, $type = 'project') { if(defined('TUTORIAL')) return $this->loadModel('tutorial')->getProject(); $project = $this->dao->select('*')->from(TABLE_PROJECT) ->where('id')->eq($projectID) ->andWhere('`type`')->in($type) ->fetch(); if(!$project) return false; if($project->end == '0000-00-00') $project->end = ''; $project = $this->loadModel('file')->replaceImgURL($project, 'desc'); return $project; } /** * Get a project by its shadow product. * * @param int $product * @access public * @return object */ public function getByShadowProduct($product) { return $this->dao->select('t2.*')->from(TABLE_PROJECTPRODUCT)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.project=t2.id') ->where('t1.product')->eq($product) ->andWhere('t2.type')->eq('project') ->limit(1) ->fetch(); } /** * Get project info. * * @param string $status * @param string $orderBy * @param int $pager * @param int $involved * @access public * @return array */ public function getInfoList($status = 'undone', $orderBy = 'order_desc', $pager = null, $involved = 0) { /* Init vars. */ $projects = $this->loadModel('program')->getProjectList(0, $status, 0, $orderBy, $pager, 0, $involved); if(empty($projects)) return array(); $projectIdList = array_keys($projects); $teams = $this->dao->select('t1.root, count(t1.id) as count')->from(TABLE_TEAM)->alias('t1') ->leftJoin(TABLE_USER)->alias('t2')->on('t1.account=t2.account') ->where('t1.root')->in($projectIdList) ->andWhere('t2.deleted')->eq(0) ->groupBy('t1.root') ->fetchAll('root'); $condition = 't2.project as project'; $estimates = $this->dao->select("$condition, sum(estimate) as estimate")->from(TABLE_TASK)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.execution = t2.id') ->where('t1.parent')->lt(1) ->andWhere('t2.project')->in($projectIdList) ->andWhere('t1.deleted')->eq(0) ->andWhere('t2.deleted')->eq(0) ->groupBy('project') ->fetchAll('project'); $this->app->loadClass('pager', $static = true); foreach($projects as $projectID => $project) { $orderBy = $project->model == 'waterfall' ? 'id_asc' : 'id_desc'; $pager = $project->model == 'waterfall' ? null : new pager(0, 1, 1); $project->executions = $this->loadModel('execution')->getStatData($projectID, 'undone', 0, 0, false, '', $orderBy, $pager); $project->teamCount = isset($teams[$projectID]) ? $teams[$projectID]->count : 0; $project->estimate = isset($estimates[$projectID]) ? round($estimates[$projectID]->estimate, 2) : 0; $project->parentName = $this->getParentProgram($project); } return $projects; } /** * Get all parent program of a program. * * @param int $parentID * @access public * @return string */ public function getParentProgram($project) { if($project->parent == 0) return ''; $parentName = $this->dao->select('id,name')->from(TABLE_PROGRAM)->where('id')->in(trim($project->path, ','))->andWhere('grade')->lt($project->grade)->orderBy('grade asc')->fetchPairs(); $parentProgram = ''; foreach($parentName as $name) $parentProgram .= $name . '/'; $parentProgram = rtrim($parentProgram, '/'); return $parentProgram; } /** * Get project overview for block. * * @param string $queryType byId|byStatus * @param string|int $param * @param string $orderBy * @param int $limit * @param string $excludedModel * @access public * @return array */ public function getOverviewList($queryType = 'byStatus', $param = 'all', $orderBy = 'id_desc', $limit = 15, $excludedModel = '') { $queryType = strtolower($queryType); $projects = $this->dao->select('*')->from(TABLE_PROJECT) ->where('type')->eq('project') ->andWhere('vision')->eq($this->config->vision) ->andWhere('deleted')->eq(0) ->beginIF($excludedModel)->andWhere('model')->ne($excludedModel)->fi() ->beginIF(!$this->app->user->admin)->andWhere('id')->in($this->app->user->view->projects)->fi() ->beginIF($queryType == 'bystatus' and $param == 'undone')->andWhere('status')->notIN('done,closed')->fi() ->beginIF($queryType == 'bystatus' and $param != 'all' and $param != 'undone')->andWhere('status')->eq($param)->fi() ->beginIF($queryType == 'byid')->andWhere('id')->eq($param)->fi() ->orderBy($orderBy) ->limit($limit) ->fetchAll('id'); if(empty($projects)) return array(); $projectIdList = array_keys($projects); $teams = $this->dao->select('t1.root, count(t1.id) as teams')->from(TABLE_TEAM)->alias('t1') ->leftJoin(TABLE_USER)->alias('t2')->on('t1.account=t2.account') ->where('t1.root')->in($projectIdList) ->andWhere('t1.type')->eq('project') ->andWhere('t2.deleted')->eq(0) ->groupBy('root')->fetchPairs(); $hours = $this->dao->select('t2.parent as project, ROUND(SUM(t1.consumed), 1) AS consumed, ROUND(SUM(t1.estimate), 1) AS estimate')->from(TABLE_TASK)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.execution = t2.id') ->where('t2.project')->in($projectIdList) ->andWhere('t2.deleted')->eq(0) ->andWhere('t1.deleted')->eq(0) ->andWhere('t1.parent')->lt(1) ->groupBy('t2.project') ->fetchAll('project'); $storySummary = $this->getTotalStoriesByProject($projectIdList); $taskSummary = $this->getTotalTaskByProject($projectIdList); $bugSummary = $this->getTotalBugByProject($projectIdList); foreach($projects as $projectID => $project) { $project->consumed = isset($hours[$projectID]) ? (float)$hours[$projectID]->consumed : 0; $project->estimate = isset($hours[$projectID]) ? (float)$hours[$projectID]->estimate : 0; $project->teamCount = isset($teams[$projectID]) ? $teams[$projectID] : 0; $project->leftBugs = isset($bugSummary[$projectID]) ? $bugSummary[$projectID]->leftBugs : 0; $project->allBugs = isset($bugSummary[$projectID]) ? $bugSummary[$projectID]->allBugs : 0; $project->doneBugs = isset($bugSummary[$projectID]) ? $bugSummary[$projectID]->doneBugs : 0; $project->allStories = isset($storySummary[$projectID]) ? $storySummary[$projectID]->allStories: 0; $project->doneStories = isset($storySummary[$projectID]) ? $storySummary[$projectID]->doneStories: 0; $project->leftStories = isset($storySummary[$projectID]) ? $storySummary[$projectID]->leftStories: 0; $project->leftTasks = isset($taskSummary[$projectID]) ? $taskSummary[$projectID]->leftTasks : 0; $project->allTasks = isset($taskSummary[$projectID]) ? $taskSummary[$projectID]->allTasks : 0; $project->waitTasks = isset($taskSummary[$projectID]) ? $taskSummary[$projectID]->waitTasks : 0; $project->doingTasks = isset($taskSummary[$projectID]) ? $taskSummary[$projectID]->doingTasks : 0; $project->rndDoneTasks = isset($taskSummary[$projectID]) ? $taskSummary[$projectID]->doneTasks : 0; $project->liteDoneTasks = isset($taskSummary[$projectID]) ? $taskSummary[$projectID]->litedoneTasks : 0; if(is_float($project->consumed)) $project->consumed = round($project->consumed, 1); if(is_float($project->estimate)) $project->estimate = round($project->estimate, 1); } return $projects; } /** * Get the number of stories associated with the project. * * @param array $projectIdList * @access public * @return int */ public function getTotalStoriesByProject($projectIdList = 0) { return $this->dao->select("t1.project, count(t2.id) as allStories, count(if(t2.status = 'active' or t2.status = 'changing', 1, null)) as leftStories, count(if(t2.status = 'closed' and t2.closedReason = 'done', 1, null)) as doneStories")->from(TABLE_PROJECTSTORY)->alias('t1') ->leftJoin(TABLE_STORY)->alias('t2')->on('t1.story=t2.id') ->where('t1.project')->in($projectIdList) ->andWhere('t2.type')->eq('story') ->andWhere('deleted')->eq('0') ->groupBy('project') ->fetchAll('project'); } /** * Get associated bugs by project. * * @param array $projectIdList * @access public * @return array */ public function getTotalBugByProject($projectIdList) { return $this->dao->select("project, count(id) as allBugs, count(if(status = 'active', 1, null)) as leftBugs, count(if(status = 'resolved', 1, null)) as doneBugs")->from(TABLE_BUG) ->where('project')->in($projectIdList) ->andWhere('deleted')->eq(0) ->groupBy('project') ->fetchAll('project'); } /** * Get associated tasks by project. * * @param array $projectIdList * @access public * @return array */ public function getTotalTaskByProject($projectIdList) { return $this->dao->select("t1.project, count(t1.id) as allTasks, count(if(t1.status = 'wait', 1, null)) as waitTasks, count(if(t1.status = 'doing', 1, null)) as doingTasks, count(if(t1.status = 'done', 1, null)) as doneTasks, count(if(t1.status = 'wait' or t1.status = 'pause' or t1.status = 'cancel', 1, null)) as leftTasks, count(if(t1.status = 'done' or t1.status = 'closed', 1, null)) as litedoneTasks")->from(TABLE_TASK)->alias('t1') ->leftJoin(TABLE_EXECUTION)->alias('t2')->on('t1.execution=t2.id') ->where('t1.project')->in($projectIdList) ->andWhere('t1.deleted')->eq(0) ->andWhere('t2.deleted')->eq(0) ->groupBy('project') ->fetchAll('project'); } /** * Get waterfall project progress. * * @param array $projectIDList * @access public * @return int */ public function getWaterfallProgress($projectIDList) { $projectList = $this->dao->select('t1.*')->from(TABLE_EXECUTION)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.project=t2.id') ->where('t1.type')->in('stage') ->andWhere('t1.deleted')->eq('0') ->andWhere('t1.vision')->eq($this->config->vision) ->andWhere('t1.project')->in($projectIDList) ->andWhere('t2.model')->eq('waterfall') ->fetchGroup('project', 'id'); $totalHour = $this->dao->select('t1.project, t1.execution, ROUND(SUM(if(t1.status !="closed" && t1.status !="cancel", `left`, 0)), 2) AS totalLeft, ROUND(SUM(consumed), 1) AS totalConsumed')->from(TABLE_TASK)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.execution = t2.id') ->where('t2.project')->in(array_keys($projectList)) ->andWhere('t1.deleted')->eq(0) ->andWhere('t1.parent')->lt(1) ->groupBy('t1.execution') ->fetchGroup('project', 'execution'); $progressList = array(); foreach($projectList as $projectID => $stageList) { $progress = 0; $projectConsumed = 0; $projectLeft = 0; foreach($stageList as $stageID => $stage) { if($stage->project != $projectID) continue; $stageTotalConsumed = isset($totalHour[$projectID][$stageID]) ? $totalHour[$projectID][$stageID]->totalConsumed : 0; $stageTotalLeft = isset($totalHour[$projectID][$stageID]) ? round($totalHour[$projectID][$stageID]->totalLeft, 1) : 0; $projectConsumed += $stageTotalConsumed; $projectLeft += $stageTotalLeft; } $progress += ($projectConsumed + $projectLeft) == 0 ? 0 : floor($projectConsumed / ($projectConsumed + $projectLeft) * 1000) / 1000 * 100; $progressList[$projectID] = $progress; } return is_numeric($projectIDList) ? (empty($progressList) ? 0 : $progressList[$projectIDList]) : $progressList; } /** * Get waterfall general PV and EV. * * @param int $projectID * @access public * @return array */ public function getWaterfallPVEVAC($projectID) { $executions = $this->dao->select('id,begin,end,realEnd,status')->from(TABLE_EXECUTION)->where('deleted')->eq(0)->andWhere('vision')->eq($this->config->vision)->andWhere('project')->eq($projectID)->fetchAll('id'); $stmt = $this->dao->select('id,status,estimate,consumed,`left`,closedReason')->from(TABLE_TASK)->where('execution')->in(array_keys($executions))->andWhere("parent")->ge(0)->andWhere("deleted")->eq(0)->andWhere('status')->ne('cancel')->query(); $PV = 0; $EV = 0; $left = 0; while($task = $stmt->fetch()) { $PV += $task->estimate; $left += $task->left; if($task->status == 'done' or $task->closedReason == 'done') { $EV += $task->estimate; } else { $task->progress = 0; if(($task->consumed + $task->left) > 0) $task->progress = round($task->consumed / ($task->consumed + $task->left) * 100, 2); $EV += round($task->estimate * $task->progress / 100, 2); } } $AC = $this->dao->select('SUM(consumed) as consumed')->from(TABLE_EFFORT) ->where('deleted')->eq(0) ->andWhere('project')->eq($projectID) ->fetch('consumed'); if(is_null($AC)) $AC = 0; return array('PV' => sprintf("%.2f", $PV), 'EV' => sprintf("%.2f", $EV), 'AC' => sprintf("%.2f", $AC), 'left' => sprintf("%.2f", $left)); } /** * Get project workhour info. * * @param int $projectID * @access public * @return object */ public function getWorkhour($projectID) { $total = $this->dao->select('ROUND(SUM(estimate), 1) AS totalEstimate, ROUND(SUM(`left`), 2) AS totalLeft')->from(TABLE_TASK)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.execution = t2.id') ->where('t2.project')->in($projectID) ->andWhere('t2.deleted')->eq(0) ->andWhere('t1.deleted')->eq(0) ->andWhere('t1.parent')->lt(1) ->fetch(); $totalConsumed = $this->dao->select('ROUND(SUM(consumed), 1) AS totalConsumed')->from(TABLE_TASK)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.execution = t2.id') ->where('t2.project')->in($projectID) ->andWhere('t2.deleted')->eq(0) ->andWhere('t1.deleted')->eq(0) ->andWhere('t1.parent')->lt(1) ->fetch('totalConsumed'); $closedTotalLeft = $this->dao->select('ROUND(SUM(`left`), 2) AS totalLeft')->from(TABLE_TASK)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.execution = t2.id') ->where('t2.project')->in($projectID) ->andWhere('t2.deleted')->eq(0) ->andWhere('t1.deleted')->eq(0) ->andWhere('t1.parent')->lt(1) ->andWhere('t1.status')->in('closed,cancel') ->fetch('totalLeft'); $workhour = new stdclass(); $workhour->totalHours = $this->dao->select('sum(t1.days * t1.hours) AS totalHours')->from(TABLE_TEAM)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.root=t2.id') ->where('t2.id')->in($projectID) ->andWhere('t2.deleted')->eq(0) ->andWhere('t1.type')->eq('project') ->fetch('totalHours'); $workhour->totalEstimate = $total->totalEstimate; $workhour->totalConsumed = $totalConsumed; $workhour->totalLeft = round($total->totalLeft - $closedTotalLeft, 1); return $workhour; } /** * Get projects consumed info. * * @param array $projectID * @param string $time * @access public * @return array */ public function getProjectsConsumed($projectIdList, $time = '') { $projects = array(); $totalConsumeds = $this->dao->select('t2.project,ROUND(SUM(t1.consumed), 1) AS totalConsumed')->from(TABLE_EFFORT)->alias('t1') ->leftJoin(TABLE_TASK)->alias('t2')->on("t1.objectID=t2.id and t1.objectType = 'task'") ->where('t2.project')->in($projectIdList) ->beginIF($time == 'THIS_YEAR')->andWhere('LEFT(t1.`date`, 4)')->eq(date('Y'))->fi() ->andWhere('t2.deleted')->eq(0) ->andWhere('t2.parent')->lt(1) ->groupBy('t2.project') ->fetchAll('project'); foreach($projectIdList as $projectID) { $project = new stdClass(); $project->totalConsumed = isset($totalConsumeds[$projectID]->totalConsumed) ? $totalConsumeds[$projectID]->totalConsumed : 0; $projects[$projectID] = $project; } return $projects; } /** * Create the link from module,method. * * @param string $module * @param string $method * @param int $projectID * @access public * @return string */ public function getProjectLink($module, $method, $projectID) { $link = helper::createLink('project', 'index', "projectID=%s"); $project = $this->dao->select('*')->from(TABLE_PROJECT)->where('id')->eq($projectID)->fetch(); if(empty($project->multiple)) return $link; if(strpos(',project,product,projectstory,story,bug,doc,testcase,testtask,testreport,repo,build,projectrelease,stakeholder,issue,risk,meeting,report,measrecord,', ',' . $module . ',') !== false) { if($module == 'project' and $method == 'execution') { $link = helper::createLink($module, $method, "status=all&projectID=%s"); } elseif($module == 'project' and strpos(',bug,testcase,testtask,testreport,build,dynamic,view,manageproducts,team,managemembers,whitelist,addwhitelist,group,', ',' . $method . ',') !== false) { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'project' and $method == 'managePriv') { $link = helper::createLink($module, 'group', "projectID=%s"); } elseif($module == 'product' and $method == 'showerrornone') { $link = helper::createLink('projectstory', 'story', "projectID=%s"); } elseif($module == 'projectstory' and $method == 'story') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'projectstory' and $method == 'linkstory') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'projectstory' and $method = 'track') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'bug') { if($method == 'create') { $link = helper::createLink($module, $method, "productID=0&branch=0&extras=projectID=%s"); } elseif($method == 'edit') { $link = helper::createLink('project', 'bug', "projectID=%s"); } } elseif($module == 'story') { if($method == 'change' or $method == 'create') { $link = helper::createLink('projectstory', 'story', "projectID=%s"); } elseif($method == 'zerocase') { $link = helper::createLink('project', 'testcase', "projectID=%s"); } } elseif($module == 'testcase') { $link = helper::createLink('project', 'testcase', "projectID=%s"); } elseif($module == 'testtask') { if($method == 'browseunits') { $link = helper::createLink('project', 'testcase', "projectID=%s"); } else { $link = helper::createLink('project', 'testtask', "projectID=%s"); } } elseif($module == 'testreport') { $link = helper::createLink('project', 'testreport', "projectID=%s"); } elseif($module == 'repo') { $link = helper::createLink($module, 'browse', "repoID=&branchID=&objectID=%s") . '#app=project'; } elseif($module == 'doc') { $link = helper::createLink($module, 'tablecontents', "type=project&objectID=%s") . '#app=project'; } elseif($module == 'build') { if($method == 'create') { $link = helper::createLink($module, $method, "executionID=&productID=&projectID=%s") . '#app=project'; } else { $link = helper::createLink('project', 'build', "projectID=%s"); } } elseif($module == 'projectrelease') { if($method == 'create') { $link = helper::createLink($module, $method, "projectID=%s"); } else { $link = helper::createLink('projectrelease', 'browse', "projectID=%s"); } } elseif($module == 'stakeholder') { if($method == 'create') { $link = helper::createLink($module, $method, "projectID=%s"); } else { $link = helper::createLink($module, 'browse', "projectID=%s"); } } elseif(strpos("issue,risk,meeting,report,measrecord", $module) !== false) { if($method == 'projectsummary') { $link = helper::createLink($module, $method, "projectID=%s") . '#app=project'; } else { $link = helper::createLink($module, 'browse', "projectID=%s"); } } } if(in_array($module, $this->config->waterfallModules)) { $link = helper::createLink($module, 'browse', "projectID=%s"); if($module == 'reviewissue') { $link = helper::createLink($module, 'issue', "projectID=%s"); } elseif($module == 'cm' and $method = 'report') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'weekly' and $method = 'index') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'milestone' and $method = 'index') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'workestimation' and $method = 'index') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'durationestimation' and $method = 'index') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'budget' and $method = 'summary') { $link = helper::createLink($module, $method, "projectID=%s"); } elseif($module == 'programplan') { $link = helper::createLink('project', 'execution', "type=all&projectID=%s"); } } return $link; } /** * Get project stat data . * * @param int $projectID * @access public * @return object */ public function getStatData($projectID) { $executions = $this->loadModel('execution')->getPairs($projectID); $storyTypeCount = $this->dao->select('count(t2.story) as storyCount,t1.type')->from(TABLE_STORY)->alias('t1') ->leftJoin(TABLE_PROJECTSTORY)->alias('t2')->on('t1.id = t2.story') ->where('t2.project')->eq($projectID) ->andWhere('t1.deleted')->eq(0) ->groupBy('t1.type') ->fetchPairs('type', 'storyCount'); $storyCount = isset($storyTypeCount['story']) ? $storyTypeCount['story'] : 0; $requirementCount = isset($storyTypeCount['requirement']) ? $storyTypeCount['requirement'] : 0; $taskCount = $this->dao->select('count(id) as taskCount')->from(TABLE_TASK)->where('execution')->in(array_keys($executions))->andWhere('deleted')->eq(0)->fetch('taskCount'); $bugCount = $this->dao->select('count(id) as bugCount')->from(TABLE_BUG)->where('project')->in($projectID)->andWhere('deleted')->eq(0)->fetch('bugCount'); $statusPairs = $this->dao->select('status,count(id) as count')->from(TABLE_TASK)->where('execution')->in(array_keys($executions))->andWhere('deleted')->eq(0)->groupBy('status')->fetchPairs('status', 'count'); $finishedCount = $this->dao->select('count(id) as taskCount')->from(TABLE_TASK)->where('execution')->in(array_keys($executions))->andWhere('finishedBy')->ne('')->andWhere('deleted')->eq(0)->fetch('taskCount'); $delayedCount = $this->dao->select('count(id) as count')->from(TABLE_TASK) ->where('execution')->in(array_keys($executions)) ->andWhere('deadline')->ne('0000-00-00') ->andWhere('deadline')->lt(helper::today()) ->andWhere('status')->in('wait,doing') ->andWhere('deleted')->eq(0) ->fetch('count'); $statData = new stdclass(); $statData->storyCount = $storyCount; $statData->requirementCount = $requirementCount; $statData->taskCount = $taskCount; $statData->bugCount = $bugCount; $statData->waitCount = zget($statusPairs, 'wait', 0); $statData->doingCount = zget($statusPairs, 'doing', 0); $statData->finishedCount = $finishedCount; $statData->delayedCount = $delayedCount; return $statData; } /** * Get project pairs. * * @access public * @return object */ public function getPairs() { return $this->dao->select('id, name')->from(TABLE_PROJECT) ->where('type')->eq('project') ->andWhere('deleted')->eq(0) ->andWhere('vision')->eq($this->config->vision) ->fetchPairs(); } /** * Get project pairs by programID. * * @param int $programID * @param status $status all|wait|doing|suspended|closed|noclosed * @param bool $isQueryAll * @param string $orderBy * @param string $excludedModel * @param string|array $model * @param string $param multiple|product * @access public * @return object */ public function getPairsByProgram($programID = '', $status = 'all', $isQueryAll = false, $orderBy = 'order_asc', $excludedModel = '', $model = '', $param = '') { if(defined('TUTORIAL')) return $this->loadModel('tutorial')->getProjectPairs(); return $this->dao->select('id, name')->from(TABLE_PROJECT) ->where('type')->eq('project') ->andWhere('deleted')->eq(0) ->andWhere('vision')->eq($this->config->vision) ->beginIF(!empty($programID))->andWhere('path')->like("%,$programID,%")->fi() ->beginIF($programID === 0)->andWhere('parent')->eq(0)->fi() ->beginIF($status != 'all' and $status != 'noclosed')->andWhere('status')->eq($status)->fi() ->beginIF($excludedModel)->andWhere('model')->ne($excludedModel)->fi() ->beginIF($model)->andWhere('model')->in($model)->fi() ->beginIF(strpos($param, 'multiple') !== false)->andWhere('multiple')->eq(1)->fi() ->beginIF(strpos($param, 'product') !== false)->andWhere('hasProduct')->eq(1)->fi() ->beginIF($status == 'noclosed')->andWhere('status')->ne('closed')->fi() ->beginIF(!$this->app->user->admin and !$isQueryAll)->andWhere('id')->in($this->app->user->view->projects)->fi() ->orderBy($orderBy) ->fetchPairs(); } /** * Get all the projects under the program set to which an project belongs. * * @param object $project * @access public * @return void */ public function getBrotherProjects($project) { if($project->parent == 0) return array($project->id => $project->id); $projectIds = array_filter(explode(',', $project->path)); $parentProgram = $this->dao->select('*')->from(TABLE_PROGRAM) ->where('id')->in($projectIds) ->andWhere('`type`')->eq('program') ->orderBy('grade desc') ->fetch(); $projects = $this->dao->select('id')->from(TABLE_PROJECT) ->where('type')->eq('project') ->andWhere('deleted')->eq(0) ->andWhere('path')->like("{$parentProgram->path}%") ->fetchPairs('id'); return $projects; } /** * Get project by id list. * * @param array $projectIdList * @param string $mode all * @access public * @return object */ public function getByIdList($projectIdList = array(), $mode = '') { return $this->dao->select('*')->from(TABLE_PROJECT) ->where('type')->eq('project') ->andWhere('id')->in($projectIdList) ->beginIF(!$this->app->user->admin and $mode != 'all')->andWhere('id')->in($this->app->user->view->projects)->fi() ->fetchAll('id'); } /** * Get project pairs by id list. * * @param array $projectIdList * @param string $model * @param string $param * @access public * @return array */ public function getPairsByIdList($projectIdList = array(), $model = '', $param = '') { return $this->dao->select('id, name')->from(TABLE_PROJECT) ->where('type')->eq('project') ->andWhere('deleted')->eq(0) ->beginIF($projectIdList)->andWhere('id')->in($projectIdList)->fi() ->beginIF(!$this->app->user->admin and $model != 'all')->andWhere('id')->in($this->app->user->view->projects)->fi() ->beginIF($model != 'all' and !empty($model))->andWhere('model')->in($model)->fi() ->beginIF(strpos($param, 'multiple') !== false)->andWhere('multiple')->eq('1')->fi() ->fetchPairs('id', 'name'); } /** * Get branches by project id. * * @param int $projectID * @access public * @return array */ public function getBranchesByProject($projectID) { return $this->dao->select('*')->from(TABLE_PROJECTPRODUCT) ->where('project')->eq($projectID) ->fetchGroup('product', 'branch'); } /** * Get branch group by project id. * * @param int $projectID * @param array|string $productIdList * @access public * @return array */ public function getBranchGroupByProject($projectID, $productIdList) { return $this->dao->select('t1.product as productID, t1.branch as branchID, t2.*')->from(TABLE_PROJECTPRODUCT)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.project=t2.id') ->where('t1.product')->in($productIdList) ->andWhere('t2.deleted')->eq(0) ->andWhere('t2.project')->eq($projectID) ->fetchGroup('productID', 'branchID'); } /** * Get No product project|execution List. * * @access public * @return array */ public function getNoProductList() { return $this->dao->select('t1.product, t2.*')->from(TABLE_PROJECTPRODUCT)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.project=t2.id') ->where('t2.hasProduct')->eq(0) ->andWhere('t2.deleted')->eq(0) ->fetchAll('id'); } /** * Get project and execution pairs. * * @access public * @return array */ public function getProjectExecutionPairs() { return $this->dao->select('project, id')->from(TABLE_PROJECT)->where('multiple')->eq('0')->andWhere('deleted')->eq('0')->fetchPairs(); } /** * Process the project privs according to the project model. * * @param string $model sprint | waterfall | noSprint * @access public * @return object */ public function processProjectPrivs($model = 'waterfall') { if($model == 'noSprint') $this->config->project->includedPriv = $this->config->project->noSprintPriv; $this->app->loadLang('group'); $privs = new stdclass(); $resource = $this->lang->resource; foreach($resource as $module => $methods) { if(!$methods) continue; if(!in_array($module, $this->config->programPriv->$model)) continue; foreach($methods as $method => $label) { if(isset($this->config->project->includedPriv[$module]) and !in_array($method, $this->config->project->includedPriv[$module])) continue; if(!isset($privs->$module)) $privs->$module = new stdclass(); $privs->$module->$method = $label; } } return $privs; } /* * Build search form. * * @param int $queryID * @param string $actionURL * * @return 0 * */ public function buildSearchForm($queryID, $actionURL) { $this->config->project->search['queryID'] = $queryID; $this->config->project->search['actionURL'] = $actionURL; $statusList = $this->lang->project->statusList; unset($statusList['delay']); $this->config->project->search['params']['status']['values'] = $statusList; $programPairs = array(0 => ''); $programPairs += $this->loadModel('program')->getPairs(); $this->config->project->search['params']['parent']['values'] = $programPairs; if(!isset($this->config->setCode) or $this->config->setCode == 0) unset($this->config->project->search['fields']['code'], $this->config->project->search['params']['code']); if($this->config->systemMode == 'light') unset($this->config->project->search['fields']['parent'], $this->config->project->search['params']['parent']); $this->loadModel('search')->setSearchParams($this->config->project->search); } /** * Build the query. * * @param int $projectID * @access public * @return object */ public function buildMenuQuery($projectID = 0) { $path = ''; $project = $this->getByID($projectID); if($project) $path = $project->path; return $this->dao->select('*')->from(TABLE_PROJECT) ->where('deleted')->eq('0') ->andWhere('type')->eq('program')->fi() ->andWhere('status')->ne('closed') ->andWhere('id')->in($this->app->user->view->programs)->fi() ->beginIF($projectID > 0)->andWhere('path')->like($path . '%')->fi() ->orderBy('grade desc, `order`') ->get(); } /** * Build project build search form. * * @param array $products * @param int $queryID * @param string $actionURL * @param string $type project|execution * @access public * @return void */ public function buildProjectBuildSearchForm($products, $queryID, $actionURL, $type = 'project') { $this->loadModel('build'); /* Set search param. */ if($type == 'execution') $this->config->build->search['module'] = 'executionBuild'; if($type == 'project') $this->config->build->search['module'] = 'projectBuild'; $this->config->build->search['actionURL'] = $actionURL; $this->config->build->search['queryID'] = $queryID; $this->config->build->search['params']['product']['values'] = $products; $this->loadModel('search')->setSearchParams($this->config->build->search); } /** * Get project pairs by model and project. * * @param string $model all|scrum|waterfall|kanban * @param int $programID * @param string $param noclosed * @param string|int|array $append * @access public * @return array */ public function getPairsByModel($model = 'all', $programID = 0, $param = '', $append = '', $orderBy = 'order_asc') { if(defined('TUTORIAL')) return $this->loadModel('tutorial')->getProjectPairs(); if($model == 'agileplus') $model = array('scrum', 'agileplus'); if($model == 'waterfallplus') $model = array('waterfall', 'waterfallplus'); $projects = $this->dao->select('id, name, path')->from(TABLE_PROJECT) ->where('type')->eq('project') ->andWhere('deleted')->eq('0') ->beginIF($programID)->andWhere('parent')->eq($programID)->fi() ->beginIF($this->config->vision)->andWhere('vision')->eq($this->config->vision)->fi() ->beginIF($model != 'all')->andWhere('model')->in($model)->fi() ->beginIF(strpos($param, 'noclosed') !== false)->andWhere('status')->ne('closed')->fi() ->beginIF(strpos($param, 'multiple') !== false)->andWhere('multiple')->eq('1')->fi() ->beginIF(!$this->app->user->admin)->andWhere('id')->in($this->app->user->view->projects)->fi() ->beginIF(!empty($append))->orWhere('id')->in($append)->fi() ->orderBy($orderBy) ->fetchAll(); $programIdList = array(); $projectProgram = array(); foreach($projects as $project) { list($programID) = explode(',', trim($project->path, ',')); $programIdList[$programID] = $programID; $projectProgram[$project->id] = $programID; } $programs = $this->dao->select('id, name')->from(TABLE_PROGRAM)->where('id')->in($programIdList)->orderBy($orderBy)->fetchPairs('id', 'name'); /* Sort by project order in the program list. */ $allProjects = array(); foreach($programs as $programID => $program) $allProjects[$programID] = array(); foreach($projects as $project) { $programID = zget($projectProgram, $project->id, ''); $allProjects[$programID][] = $project; } $pairs = array(); foreach($allProjects as $programID => $projects) { foreach($projects as $project) { $projectName = $project->name; if($this->config->systemMode == 'ALM') { $programID = zget($projectProgram, $project->id, ''); if($programID != $project->id) $projectName = zget($programs, $programID, '') . ' / ' . $projectName; } $pairs[$project->id] = $projectName; } } return $pairs; } /** * Get stories by project id. * * @param int $projectID * @access public * @return array */ public function getStoriesByProject($projectID = 0) { return $this->dao->select("t2.product, t2.branch, GROUP_CONCAT(t2.story) as storyIDList")->from(TABLE_STORY)->alias('t1') ->leftJoin(TABLE_PROJECTSTORY)->alias('t2')->on('t1.id=t2.story') ->where('t1.deleted')->eq(0) ->beginIF($projectID)->andWhere('t2.project')->eq($projectID)->fi() ->groupBy('product, branch') ->fetchGroup('product', 'branch'); } /** * Get the tree menu of project. * * @param int $projectID * @param string $userFunc * @param int $param * @access public * @return string */ public function getTreeMenu($projectID = 0, $userFunc = '', $param = 0) { $projectMenu = array(); $stmt = $this->dbh->query($this->buildMenuQuery($projectID)); while($project = $stmt->fetch()) { $linkHtml = call_user_func($userFunc, $project, $param); if(isset($projectMenu[$project->id]) and !empty($projectMenu[$project->id])) { if(!isset($projectMenu[$project->parent])) $projectMenu[$project->parent] = ''; $projectMenu[$project->parent] .= "
  • $linkHtml"; $projectMenu[$project->parent] .= "\n"; } else { if(isset($projectMenu[$project->parent]) and !empty($projectMenu[$project->parent])) { $projectMenu[$project->parent] .= "
  • $linkHtml\n"; } else { $projectMenu[$project->parent] = "
  • $linkHtml\n"; } } $projectMenu[$project->parent] .= "
  • \n"; } krsort($projectMenu); $projectMenu = array_pop($projectMenu); $lastMenu = "\n"; return $lastMenu; } /** * Create the manage link. * * @param object $project * @access public * @return string */ public function createManageLink($project) { $link = $project->type == 'program' ? helper::createLink('project', 'browse', "projectID={$project->id}&status=all") : helper::createLink('project', 'index', "projectID={$project->id}", '', '', $project->id); if($this->app->rawModule == 'execution') $link = helper::createLink('execution', 'all', "status=all&projectID={$project->id}"); return html::a($link, $project->name, '_self', "id=program{$project->id} title='{$project->name}' class='text-ellipsis'"); } /** * Create a project. * * @access public * @return int|bool */ public function create() { $project = fixer::input('post') ->callFunc('name', 'trim') ->setDefault('status', 'wait') ->setIF($this->post->delta == 999, 'end', LONG_TIME) ->setIF($this->post->delta == 999, 'days', 0) ->setIF($this->post->acl == 'open', 'whitelist', '') ->setIF(!isset($_POST['whitelist']), 'whitelist', '') ->setIF(!isset($_POST['multiple']), 'multiple', '1') ->setDefault('openedBy', $this->app->user->account) ->setDefault('openedDate', helper::now()) ->setDefault('team', $this->post->name) ->setDefault('lastEditedBy', $this->app->user->account) ->setDefault('lastEditedDate', helper::now()) ->setDefault('days', '0') ->add('type', 'project') ->join('whitelist', ',') ->stripTags($this->config->project->editor->create['id'], $this->config->allowedTags) ->remove('products,branch,plans,delta,newProduct,productName,future,contactListMenu,teamMembers') ->get(); if(!isset($this->config->setCode) or $this->config->setCode == 0) unset($project->code); /* Lean mode relation defaultProgram. */ if($this->config->systemMode == 'light') $project->parent = $this->config->global->defaultProgram; $linkedProductsCount = 0; if($project->hasProduct && isset($_POST['products'])) { foreach($_POST['products'] as $product) { if(!empty($product)) $linkedProductsCount++; } } if($_POST['products']) { $topProgramID = $this->loadModel('program')->getTopByID($project->parent); $multipleProducts = $this->loadModel('product')->getMultiBranchPairs($topProgramID); foreach($_POST['products'] as $index => $productID) { if(isset($multipleProducts[$productID]) and empty($_POST['branch'][$index])) { dao::$errors[] = $this->lang->project->emptyBranch; return false; } } } $program = new stdClass(); if($project->parent) { $program = $this->dao->select('*')->from(TABLE_PROGRAM)->where('id')->eq($project->parent)->fetch(); /* Judge products not empty. */ if($project->hasProduct && empty($linkedProductsCount) and !isset($_POST['newProduct'])) { dao::$errors['products0'] = $this->lang->project->productNotEmpty; return false; } } /* Judge workdays is legitimate. */ $workdays = helper::diffDate($project->end, $project->begin) + 1; if(isset($project->days) and $project->days > $workdays) { dao::$errors['days'] = sprintf($this->lang->project->workdaysExceed, $workdays); return false; } if(!empty($project->budget)) { if(!is_numeric($project->budget)) { dao::$errors['budget'] = sprintf($this->lang->project->budgetNumber); return false; } else if(is_numeric($project->budget) and ($project->budget < 0)) { dao::$errors['budget'] = sprintf($this->lang->project->budgetGe0); return false; } else { $project->budget = round((float)$this->post->budget, 2); } } /* When select create new product, product name cannot be empty and duplicate. */ if($project->hasProduct && isset($_POST['newProduct'])) { if(empty($_POST['productName'])) { $this->app->loadLang('product'); dao::$errors['productName'] = sprintf($this->lang->error->notempty, $this->lang->product->name); return false; } else { $programID = isset($project->parent) ? $project->parent : 0; $existProductName = $this->dao->select('name')->from(TABLE_PRODUCT)->where('name')->eq($_POST['productName'])->andWhere('program')->eq($programID)->fetch('name'); if(!empty($existProductName)) { dao::$errors['productName'] = $this->lang->project->existProductName; return false; } } } $requiredFields = $this->config->project->create->requiredFields; if($this->post->delta == 999) $requiredFields = trim(str_replace(',end,', ',', ",{$requiredFields},"), ','); $this->lang->error->unique = $this->lang->error->repeat; $project = $this->loadModel('file')->processImgURL($project, $this->config->project->editor->create['id'], $this->post->uid); $this->dao->insert(TABLE_PROJECT)->data($project) ->autoCheck() ->batchcheck($requiredFields, 'notempty') ->checkIF(!empty($project->name), 'name', 'unique', "`type`='project' and `parent` = $project->parent and `model` = '{$project->model}' and `deleted` = '0'") ->checkIF(!empty($project->code), 'code', 'unique', "`type`='project' and `model` = '{$project->model}' and `deleted` = '0'") ->checkIF($project->end != '', 'end', 'gt', $project->begin) ->checkFlow() ->exec(); /* Add the creater to the team. */ if(!dao::isError()) { $projectID = $this->dao->lastInsertId(); /* Set team of project. */ $members = isset($_POST['teamMembers']) ? $_POST['teamMembers'] : array(); array_push($members, $project->PM, $project->openedBy); $members = array_unique($members); $roles = $this->loadModel('user')->getUserRoles(array_values($members)); $teamMembers = array(); foreach($members as $account) { if(empty($account)) continue; $member = new stdClass(); $member->root = $projectID; $member->type = 'project'; $member->account = $account; $member->role = zget($roles, $account, ''); $member->join = helper::now(); $member->days = zget($project, 'days', 0); $member->hours = $this->config->execution->defaultWorkhours; $this->dao->insert(TABLE_TEAM)->data($member)->exec(); $teamMembers[$account] = $member; } $this->loadModel('execution')->addProjectMembers($projectID, $teamMembers); $whitelist = explode(',', $project->whitelist); $this->loadModel('personnel')->updateWhitelist($whitelist, 'project', $projectID); /* Create doc lib. */ $this->app->loadLang('doc'); $authorizedUsers = array(); if($project->parent and $project->acl == 'program') { $stakeHolders = $this->loadModel('stakeholder')->getStakeHolderPairs($project->parent); $authorizedUsers = array_keys($stakeHolders); foreach(explode(',', $project->whitelist) as $user) { if(empty($user)) continue; $authorizedUsers[$user] = $user; } $authorizedUsers[$project->PM] = $project->PM; $authorizedUsers[$project->openedBy] = $project->openedBy; $authorizedUsers[$program->PM] = $program->PM; $authorizedUsers[$program->openedBy] = $program->openedBy; } $lib = new stdclass(); $lib->project = $projectID; $lib->name = $this->lang->doclib->main['project']; $lib->type = 'project'; $lib->main = '1'; $lib->acl = $project->acl != 'program' ? $project->acl : 'custom'; $lib->users = ',' . implode(',', array_filter($authorizedUsers)) . ','; $lib->vision = zget($project, 'vision', 'rnd'); $this->dao->insert(TABLE_DOCLIB)->data($lib)->exec(); if($project->hasProduct) $this->updateProducts($projectID); if(!$project->hasProduct or isset($_POST['newProduct']) or (!$project->parent and empty($linkedProductsCount))) { /* If parent not empty, link products or create products. */ $product = new stdclass(); $product->name = $project->hasProduct && $this->post->productName ? $this->post->productName : $project->name; $product->shadow = zget($project, 'vision', 'rnd') == 'rnd' ? (int)empty($project->hasProduct) : 1; $product->bind = $this->post->parent ? 0 : 1; $product->program = $project->parent ? current(array_filter(explode(',', $program->path))) : 0; $product->acl = $project->acl == 'open' ? 'open' : 'private'; $product->PO = $project->PM; $product->createdBy = $this->app->user->account; $product->createdDate = helper::now(); $product->status = 'normal'; $product->createdVersion = $this->config->version; $product->vision = zget($project, 'vision', 'rnd'); $this->dao->insert(TABLE_PRODUCT)->data($product)->exec(); $productID = $this->dao->lastInsertId(); if(!$project->hasProduct) $this->loadModel('personnel')->updateWhitelist($whitelist, 'product', $productID); $this->loadModel('action')->create('product', $productID, 'opened'); $this->dao->update(TABLE_PRODUCT)->set('`order`')->eq($productID * 5)->where('id')->eq($productID)->exec(); if($product->acl != 'open') $this->loadModel('user')->updateUserView($productID, 'product'); $projectProduct = new stdclass(); $projectProduct->project = $projectID; $projectProduct->product = $productID; $this->dao->insert(TABLE_PROJECTPRODUCT)->data($projectProduct)->exec(); if($project->hasProduct) { /* Create doc lib. */ $this->app->loadLang('doc'); $lib = new stdclass(); $lib->product = $productID; $lib->name = $this->lang->doclib->main['product']; $lib->type = 'product'; $lib->main = '1'; $lib->acl = 'default'; $this->dao->insert(TABLE_DOCLIB)->data($lib)->exec(); } } /* Save order. */ $this->dao->update(TABLE_PROJECT)->set('`order`')->eq($projectID * 5)->where('id')->eq($projectID)->exec(); $this->file->updateObjectID($this->post->uid, $projectID, 'project'); $this->loadModel('program')->setTreePath($projectID); /* Add project admin. */ $groupPriv = $this->dao->select('t1.*')->from(TABLE_USERGROUP)->alias('t1') ->leftJoin(TABLE_GROUP)->alias('t2')->on('t1.`group` = t2.id') ->where('t1.account')->eq($this->app->user->account) ->andWhere('t2.role')->eq('projectAdmin') ->fetch(); if(!empty($groupPriv)) { $newProject = $groupPriv->project . ",$projectID"; $this->dao->update(TABLE_USERGROUP)->set('project')->eq($newProject)->where('account')->eq($groupPriv->account)->andWhere('`group`')->eq($groupPriv->group)->exec(); } else { $projectAdminID = $this->dao->select('id')->from(TABLE_GROUP)->where('role')->eq('projectAdmin')->fetch('id'); $groupPriv = new stdclass(); $groupPriv->account = $this->app->user->account; $groupPriv->group = $projectAdminID; $groupPriv->project = $projectID; $this->dao->replace(TABLE_USERGROUP)->data($groupPriv)->exec(); } if($project->acl != 'open') $this->loadModel('user')->updateUserView($projectID, 'project'); if(empty($project->multiple) and $project->model != 'waterfall' and $project->model != 'waterfallplus') $this->loadModel('execution')->createDefaultSprint($projectID); return $projectID; } } /** * Update project. * * @param int $projectID * @access public * @return array */ public function update($projectID = 0) { $oldProject = $this->dao->findById($projectID)->from(TABLE_PROJECT)->fetch(); $linkedProducts = $this->dao->select('product')->from(TABLE_PROJECTPRODUCT)->where('project')->eq($projectID)->fetchPairs(); $_POST['products'] = isset($_POST['products']) ? array_filter($_POST['products']) : $linkedProducts; $project = fixer::input('post') ->add('id', $projectID) ->callFunc('name', 'trim') ->setDefault('team', $this->post->name) ->setDefault('lastEditedBy', $this->app->user->account) ->setDefault('lastEditedDate', helper::now()) ->setDefault('days', '0') ->setIF($this->post->delta == 999, 'end', LONG_TIME) ->setIF($this->post->delta == 999, 'days', 0) ->setIF($this->post->begin == '0000-00-00', 'begin', '') ->setIF($this->post->end == '0000-00-00', 'end', '') ->setIF($this->post->future, 'budget', 0) ->setIF($this->post->budget != 0, 'budget', round((float)$this->post->budget, 2)) ->setIF(!isset($_POST['whitelist']), 'whitelist', '') ->join('whitelist', ',') ->stripTags($this->config->project->editor->edit['id'], $this->config->allowedTags) ->remove('products,branch,plans,delta,future,contactListMenu,teamMembers') ->get(); if(!isset($project->parent)) $project->parent = $oldProject->parent; $executionsCount = $this->dao->select('COUNT(*) as count')->from(TABLE_PROJECT)->where('project')->eq($project->id)->andWhere('deleted')->eq('0')->fetch('count'); if(!empty($executionsCount) and $oldProject->multiple) { $minExecutionBegin = $this->dao->select('begin as minBegin')->from(TABLE_PROJECT)->where('project')->eq($project->id)->andWhere('deleted')->eq('0')->orderBy('begin_asc')->fetch(); $maxExecutionEnd = $this->dao->select('end as maxEnd')->from(TABLE_PROJECT)->where('project')->eq($project->id)->andWhere('deleted')->eq('0')->orderBy('end_desc')->fetch(); if($minExecutionBegin and $project->begin > $minExecutionBegin->minBegin) dao::$errors['begin'] = sprintf($this->lang->project->begigLetterExecution, $minExecutionBegin->minBegin); if($maxExecutionEnd and $project->end < $maxExecutionEnd->maxEnd) dao::$errors['end'] = sprintf($this->lang->project->endGreateExecution, $maxExecutionEnd->maxEnd); if(dao::isError()) return false; } if($_POST['products']) { $topProgramID = $this->loadModel('program')->getTopByID($project->parent); $multipleProducts = $this->loadModel('product')->getMultiBranchPairs($topProgramID); foreach($_POST['products'] as $index => $productID) { if(isset($multipleProducts[$productID]) and empty($_POST['branch'][$index])) { dao::$errors[] = $this->lang->project->emptyBranch; return false; } } } /* Judge products not empty. */ $linkedProductsCount = 0; foreach($_POST['products'] as $product) { if(!empty($product)) $linkedProductsCount++; } if(empty($linkedProductsCount)) { dao::$errors[] = $this->lang->project->errorNoProducts; return false; } /* Judge workdays is legitimate. */ $workdays = helper::diffDate($project->end, $project->begin) + 1; if(isset($project->days) and $project->days > $workdays) { dao::$errors['days'] = sprintf($this->lang->project->workdaysExceed, $workdays); return false; } $project = $this->loadModel('file')->processImgURL($project, $this->config->project->editor->edit['id'], $this->post->uid); $requiredFields = $this->config->project->edit->requiredFields; if($this->post->delta == 999) $requiredFields = trim(str_replace(',end,', ',', ",{$requiredFields},"), ','); $this->lang->error->unique = $this->lang->error->repeat; $this->dao->update(TABLE_PROJECT)->data($project) ->autoCheck($skipFields = 'begin,end') ->batchcheck($requiredFields, 'notempty') ->checkIF($project->begin != '', 'begin', 'date') ->checkIF($project->end != '', 'end', 'date') ->checkIF($project->end != '', 'end', 'gt', $project->begin) ->checkIF(!empty($project->name), 'name', 'unique', "id != $projectID and `type` = 'project' and `parent` = '$oldProject->parent' and `model` = '{$project->model}' and `deleted` = '0'") ->checkIF(!empty($project->code), 'code', 'unique', "id != $projectID and `type` = 'project' and `model` = '{$project->model}' and `deleted` = '0'") ->checkFlow() ->where('id')->eq($projectID) ->exec(); if(dao::isError()) return false; if(!$oldProject->hasProduct and ($oldProject->name != $project->name or $oldProject->parent != $project->parent or $oldProject->acl != $project->acl)) $this->updateShadowProduct($project); /* Get team and language item. */ $this->loadModel('user'); $team = $this->user->getTeamMemberPairs($projectID, 'project'); $members = isset($_POST['teamMembers']) ? $_POST['teamMembers'] : array(); array_push($members, $project->PM); $members = array_unique($members); $roles = $this->user->getUserRoles(array_values($members)); $teamMembers = array(); foreach($members as $account) { if(empty($account) or isset($team[$account])) continue; $member = new stdclass(); $member->root = (int)$projectID; $member->account = $account; $member->join = helper::today(); $member->role = zget($roles, $account, ''); $member->days = zget($project, 'days', 0); $member->type = 'project'; $member->hours = $this->config->execution->defaultWorkhours; $this->dao->replace(TABLE_TEAM)->data($member)->exec(); $teamMembers[$account] = $member; } if($oldProject->model == 'kanban') { $this->dao->delete()->from(TABLE_TEAM) ->where('root')->eq((int)$projectID) ->andWhere('type')->eq('project') ->andWhere('account')->in(array_keys($team)) ->andWhere('account')->notin(array_values($members)) ->andWhere('account')->ne($oldProject->openedBy) ->exec(); } if(!empty($projectID) and !empty($teamMembers)) $this->loadModel('execution')->addProjectMembers($projectID, $teamMembers); if(!dao::isError()) { $this->updateProducts($projectID, $_POST['products']); if(empty($oldProject->division)) { $executions = $this->loadModel('execution')->getPairs($projectID); foreach(array_keys($executions) as $executionID) $this->execution->updateProducts($executionID); } $this->file->updateObjectID($this->post->uid, $projectID, 'project'); $whitelist = explode(',', $project->whitelist); $this->loadModel('personnel')->updateWhitelist($whitelist, 'project', $projectID); if(!$oldProject->hasProduct) $this->loadModel('personnel')->updateWhitelist($whitelist, 'product', current($linkedProducts)); if($project->acl != 'open') { $this->loadModel('user')->updateUserView($projectID, 'project'); $executions = $this->dao->select('id')->from(TABLE_EXECUTION)->where('project')->eq($projectID)->fetchPairs('id', 'id'); if($executions) $this->user->updateUserView($executions, 'sprint'); } if($oldProject->parent != $project->parent) $this->loadModel('program')->processNode($projectID, $project->parent, $oldProject->path, $oldProject->grade); /* Fix whitelist changes. */ $oldWhitelist = explode(',', $oldProject->whitelist) and $oldWhitelist = array_filter($oldWhitelist); $newWhitelist = explode(',', $project->whitelist) and $newWhitelist = array_filter($newWhitelist); if(count($oldWhitelist) == count($newWhitelist) and count(array_diff($oldWhitelist, $newWhitelist)) == 0) unset($project->whitelist); /* Add linkedproducts changes. */ $oldProject->linkedProducts = implode(',', $linkedProducts); $project->linkedProducts = implode(',', $_POST['products']); $unlinkedProducts = array_diff($linkedProducts, $_POST['products']); if(!empty($unlinkedProducts)) { $products = $this->dao->select('name')->from(TABLE_PRODUCT)->where('id')->in($unlinkedProducts)->fetchPairs(); $this->loadModel('action')->create('project', $projectID, 'unlinkproduct', '', implode(',', $products)); } /* Activate or close the shadow product of the project. */ if(!$oldProject->hasProduct && $oldProject->status != $project->status && strpos('doing,closed', $project->status) !== false) { $productID = $this->loadModel('product')->getProductIDByProject($projectID); if($project->status == 'doing') $this->product->activate($productID); if($project->status == 'closed') $this->product->close($productID); } if(empty($oldProject->multiple) and $oldProject->model != 'waterfall') $this->loadModel('execution')->syncNoMultipleSprint($projectID); return common::createChanges($oldProject, $project); } } /** * Batch update projects. * * @access public * @return array */ public function batchUpdate() { $projects = array(); $allChanges = array(); $data = fixer::input('post')->get(); $oldProjects = $this->getByIdList($this->post->projectIdList); $nameList = array(); $extendFields = $this->getFlowExtendFields(); foreach($data->projectIdList as $projectID) { $projectID = (int)$projectID; $projectName = $data->names[$projectID]; if(isset($data->codes)) $projectCode = $data->codes[$projectID]; $projects[$projectID] = new stdClass(); if(isset($data->parents[$projectID])) $projects[$projectID]->parent = $data->parents[$projectID]; $projects[$projectID]->id = $projectID; $projects[$projectID]->name = $projectName; $projects[$projectID]->model = $oldProjects[$projectID]->model; $projects[$projectID]->PM = $data->PMs[$projectID]; $projects[$projectID]->begin = $data->begins[$projectID]; $projects[$projectID]->end = $data->ends[$projectID] == $this->lang->project->longTime ? LONG_TIME : $data->ends[$projectID]; $projects[$projectID]->days = $data->ends[$projectID] == $this->lang->project->longTime ? 0 : $data->dayses[$projectID]; $projects[$projectID]->acl = $data->acls[$projectID]; $projects[$projectID]->lastEditedBy = $this->app->user->account; $projects[$projectID]->lastEditedDate = helper::now(); if(isset($data->codes)) $projects[$projectID]->code = $projectCode; foreach($extendFields as $extendField) { $projects[$projectID]->{$extendField->field} = $this->post->{$extendField->field}[$projectID]; if(is_array($projects[$projectID]->{$extendField->field})) $projects[$projectID]->{$extendField->field} = join(',', $projects[$projectID]->{$extendField->field}); $projects[$projectID]->{$extendField->field} = htmlSpecialString($projects[$projectID]->{$extendField->field}); } } if(dao::isError()) return false; $this->loadModel('execution'); $this->lang->error->unique = $this->lang->error->repeat; foreach($projects as $projectID => $project) { $oldProject = $oldProjects[$projectID]; $parentID = !isset($project->parent) ? $oldProject->parent : $project->parent; $this->dao->update(TABLE_PROJECT)->data($project) ->autoCheck($skipFields = 'begin,end') ->batchCheck($this->config->project->edit->requiredFields, 'notempty') ->checkIF($project->begin != '', 'begin', 'date') ->checkIF($project->end != '', 'end', 'date') ->checkIF($project->end != '', 'end', 'gt', $project->begin) ->checkIF(!empty($project->name), 'name', 'unique', "id != $projectID and `type`='project' and `parent` = $parentID and `model` = '{$project->model}' and `deleted` = '0'") ->checkIF(!empty($project->code), 'code', 'unique', "id != $projectID and `type`='project' and `model` = '{$project->model}' and `deleted` = '0'") ->checkFlow() ->where('id')->eq($projectID) ->exec(); if(dao::isError()) { $errors = dao::getError(); foreach($errors as $key => $error) dao::$errors[$key][0] = 'ID' . $projectID . $error[0]; return false; } if(!dao::isError()) { if(!$oldProject->hasProduct and ($oldProject->name != $project->name or $oldProject->parent != $project->parent or $oldProject->acl != $project->acl)) $this->updateShadowProduct($project); if(isset($project->parent)) { $linkedProducts = $this->dao->select('product')->from(TABLE_PROJECTPRODUCT)->where('project')->eq($projectID)->fetchPairs(); $this->updateProductProgram($oldProject->parent, $project->parent, $linkedProducts); if($oldProject->parent != $project->parent) $this->loadModel('program')->processNode($projectID, $project->parent, $oldProject->path, $oldProject->grade); } /* When acl is open, white list set empty. When acl is private,update user view. */ if($project->acl == 'open') $this->loadModel('personnel')->updateWhitelist(array(), 'project', $projectID); if($project->acl != 'open') $this->loadModel('user')->updateUserView($projectID, 'project'); $this->executeHooks($projectID); if(empty($oldProject->multiple) and $oldProject->model != 'waterfall') $this->execution->syncNoMultipleSprint($projectID); } $allChanges[$projectID] = common::createChanges($oldProject, $project); } return $allChanges; } /** * Start project. * * @param int $projectID * @param string $type * @access public * @return array */ public function start($projectID, $type = 'project') { $oldProject = $this->getById($projectID, $type); $now = helper::now(); $editorIdList = $this->config->project->editor->start['id']; if($this->app->rawModule == 'program') $editorIdList = $this->config->program->editor->start['id']; $project = fixer::input('post') ->add('id', $projectID) ->setDefault('status', 'doing') ->setDefault('lastEditedBy', $this->app->user->account) ->setDefault('lastEditedDate', $now) ->stripTags($editorIdList, $this->config->allowedTags) ->remove('comment')->get(); $project = $this->loadModel('file')->processImgURL($project, $editorIdList, $this->post->uid); $this->dao->update(TABLE_PROJECT)->data($project) ->autoCheck() ->check($this->config->project->start->requiredFields, 'notempty') ->checkIF($project->realBegan != '', 'realBegan', 'le', helper::today()) ->checkFlow() ->where('id')->eq((int)$projectID) ->exec(); /* When it has multiple errors, only the first one is prompted */ if(dao::isError() and count(dao::$errors['realBegan']) > 1) dao::$errors['realBegan'] = dao::$errors['realBegan'][0]; if(!dao::isError()) { if(!$oldProject->multiple) $this->changeExecutionStatus($projectID, 'start'); return common::createChanges($oldProject, $project); } } /** * Put project off. * * @param int $projectID * @access public * @return void */ public function putoff($projectID) { $oldProject = $this->getById($projectID); $now = helper::now(); $project = fixer::input('post') ->add('id', $projectID) ->setDefault('lastEditedBy', $this->app->user->account) ->setDefault('lastEditedDate', $now) ->remove('comment') ->get(); $this->dao->update(TABLE_PROJECT)->data($project) ->autoCheck() ->checkFlow() ->where('id')->eq((int)$projectID) ->exec(); if(!dao::isError()) return common::createChanges($oldProject, $project); } /** * Suspend project. * * @param int $projectID * @access public * @return void */ public function suspend($projectID) { $editorIdList = $this->config->project->editor->suspend['id']; if($this->app->rawModule == 'program') $editorIdList = $this->config->program->editor->suspend['id']; $oldProject = $this->getById($projectID); $project = fixer::input('post') ->add('id', $projectID) ->setDefault('status', 'suspended') ->setDefault('lastEditedBy', $this->app->user->account) ->setDefault('lastEditedDate', helper::now()) ->setDefault('suspendedDate', helper::today()) ->stripTags($editorIdList, $this->config->allowedTags) ->remove('comment')->get(); $project = $this->loadModel('file')->processImgURL($project, $editorIdList, $this->post->uid); $this->dao->update(TABLE_PROJECT)->data($project) ->autoCheck() ->checkFlow() ->where('id')->eq((int)$projectID) ->exec(); if(!dao::isError()) { if(!$oldProject->multiple) $this->changeExecutionStatus($projectID, 'suspend'); return common::createChanges($oldProject, $project); } } /** * Activate project. * * @param int $projectID * @access public * @return void */ public function activate($projectID) { $oldProject = $this->getById($projectID); $now = helper::now(); $editorIdList = $this->config->project->editor->activate['id']; if($this->app->rawModule == 'program') $editorIdList = $this->config->program->editor->activate['id']; $project = fixer::input('post') ->add('id', $projectID) ->setDefault('realEnd','') ->setDefault('status', 'doing') ->setDefault('lastEditedBy', $this->app->user->account) ->setDefault('lastEditedDate', $now) ->setIF($oldProject->realBegan == '0000-00-00', 'realBegan', helper::today()) ->stripTags($editorIdList, $this->config->allowedTags) ->remove('comment,readjustTime,readjustTask') ->get(); if(!$this->post->readjustTime) { unset($project->begin); unset($project->end); } $project = $this->loadModel('file')->processImgURL($project, $editorIdList, $this->post->uid); $this->dao->update(TABLE_PROJECT)->data($project) ->autoCheck() ->checkFlow() ->where('id')->eq((int)$projectID) ->exec(); if(dao::isError()) return false; if(empty($oldProject->multiple) and $oldProject->model != 'waterfall') $this->loadModel('execution')->syncNoMultipleSprint($projectID); /* Readjust task. */ if($this->post->readjustTime and $this->post->readjustTask) { $beginTimeStamp = strtotime($project->begin); $tasks = $this->dao->select('id,estStarted,deadline,status')->from(TABLE_TASK) ->where('deadline')->ne('0000-00-00') ->andWhere('status')->in('wait,doing') ->andWhere('project')->eq($projectID) ->fetchAll(); foreach($tasks as $task) { if($task->status == 'wait' and !helper::isZeroDate($task->estStarted)) { $taskDays = helper::diffDate($task->deadline, $task->estStarted); $taskOffset = helper::diffDate($task->estStarted, $oldProject->begin); $estStartedTimeStamp = $beginTimeStamp + $taskOffset * 24 * 3600; $estStarted = date('Y-m-d', $estStartedTimeStamp); $deadline = date('Y-m-d', $estStartedTimeStamp + $taskDays * 24 * 3600); if($estStarted > $project->end) $estStarted = $project->end; if($deadline > $project->end) $deadline = $project->end; $this->dao->update(TABLE_TASK)->set('estStarted')->eq($estStarted)->set('deadline')->eq($deadline)->where('id')->eq($task->id)->exec(); } else { $taskOffset = helper::diffDate($task->deadline, $oldProject->begin); $deadline = date('Y-m-d', $beginTimeStamp + $taskOffset * 24 * 3600); if($deadline > $project->end) $deadline = $project->end; $this->dao->update(TABLE_TASK)->set('deadline')->eq($deadline)->where('id')->eq($task->id)->exec(); } } } /* Activate the shadow product of the project. */ if(!$oldProject->hasProduct) { $productID = $this->loadModel('product')->getProductIDByProject($projectID); $this->product->activate($productID); } return common::createChanges($oldProject, $project); } /** * Close project. * * @param int $projectID * @access public * @return array */ public function close($projectID) { $oldProject = $this->getById($projectID); $now = helper::now(); $editorIdList = $this->config->project->editor->close['id']; if($this->app->rawModule == 'program') $editorIdList = $this->config->program->editor->close['id']; $project = fixer::input('post') ->add('id', $projectID) ->setDefault('status', 'closed') ->setDefault('closedBy', $this->app->user->account) ->setDefault('closedDate', $now) ->setDefault('lastEditedBy', $this->app->user->account) ->setDefault('lastEditedDate', $now) ->stripTags($editorIdList, $this->config->allowedTags) ->remove('comment') ->get(); $this->lang->error->ge = $this->lang->project->ge; $project = $this->loadModel('file')->processImgURL($project, $editorIdList, $this->post->uid); $this->dao->update(TABLE_PROJECT)->data($project) ->autoCheck() ->check($this->config->project->close->requiredFields, 'notempty') ->checkIF($project->realEnd != '', 'realEnd', 'le', helper::today()) ->checkIF($project->realEnd != '', 'realEnd', 'ge', $oldProject->realBegan) ->checkFlow() ->where('id')->eq((int)$projectID) ->exec(); /* When it has multiple errors, only the first one is prompted */ if(dao::isError()) { if(count(dao::$errors['realEnd']) > 1) dao::$errors['realEnd'] = dao::$errors['realEnd'][0]; return false; } if(!$oldProject->multiple) $this->changeExecutionStatus($projectID, 'close'); /* Close the shadow product of the project. */ if(!$oldProject->hasProduct) { $productID = $this->loadModel('product')->getProductIDByProject($projectID); unset($_POST); $this->product->close($productID); } $this->loadModel('score')->create('project', 'close', $oldProject); return common::createChanges($oldProject, $project); } /** * Modify the execution status when changing the status of no execution project. * * @param int $projectID * @param string $status * @access public * @return array */ public function changeExecutionStatus($projectID, $status) { if(!in_array($status, array('start', 'suspend', 'activate', 'close'))) return false; $executionID = $this->dao->select('id')->from(TABLE_EXECUTION)->where('project')->eq($projectID)->andWhere('multiple')->eq('0')->fetch('id'); if(!$executionID) return false; return $this->loadModel('execution')->$status($executionID); } /** * Update shadow product for its project. * * @param object $project * @access public * @return bool */ public function updateShadowProduct($project) { $product = $this->dao->select('product')->from(TABLE_PROJECTPRODUCT)->where('project')->eq($project->id)->fetch('product'); $topProgram = !empty($project->parent) ? $this->loadModel('program')->getTopByID($project->parent) : 0; $this->dao->update(TABLE_PRODUCT)->set('name')->eq($project->name)->set('program')->eq($topProgram)->set('acl')->eq($project->acl)->where('id')->eq($product)->exec(); return !dao::isError(); } /** * Update the program of the product. * * @param int $oldProgram * @param int $newProgram * @param array $products * @access public * @return void */ public function updateProductProgram($oldProgram, $newProgram, $products) { $this->loadModel('action'); $this->loadModel('program'); /* Product belonging project set processing. */ $oldTopProgram = $this->program->getTopByID($oldProgram); $newTopProgram = $this->program->getTopByID($newProgram); if($oldTopProgram != $newTopProgram) { foreach($products as $productID) { $oldProduct = $this->dao->findById($productID)->from(TABLE_PRODUCT)->fetch(); $this->dao->update(TABLE_PRODUCT)->set('program')->eq((int)$newTopProgram)->where('id')->eq((int)$productID)->exec(); $newProduct = $this->dao->findById($productID)->from(TABLE_PRODUCT)->fetch(); $changes = common::createChanges($oldProduct, $newProduct); $actionID = $this->action->create('product', $productID, 'edited'); $this->action->logHistory($actionID, $changes); } } } /** * Unlink a member. * * @param int $projectID * @param string $account * @param string $removeExecution no|yes * @access public * @return void */ public function unlinkMember($projectID, $account, $removeExecution = 'no') { $this->dao->delete()->from(TABLE_TEAM)->where('root')->eq((int)$projectID)->andWhere('type')->eq('project')->andWhere('account')->eq($account)->exec(); $this->loadModel('user')->updateUserView($projectID, 'project', array($account)); if($removeExecution == 'yes') { $executions = $this->loadModel('execution')->getByProject($projectID, 'undone', 0, true); $this->dao->delete()->from(TABLE_TEAM)->where('root')->in(array_keys($executions))->andWhere('type')->eq('execution')->andWhere('account')->eq($account)->exec(); $this->user->updateUserView(array_keys($executions), 'sprint', array($account)); } $linkedProducts = $this->loadModel('product')->getProductPairsByProject($projectID); if(!empty($linkedProducts)) $this->user->updateUserView(array_keys($linkedProducts), 'product', array($account)); } /** * Manage team members. * * @param int $projectID * @access public * @return void */ public function manageMembers($projectID) { $project = $this->getByID($projectID); $data = (array)fixer::input('post')->get(); extract($data); $projectID = (int)$projectID; $projectType = 'project'; $accounts = array_unique($accounts); $oldJoin = $this->dao->select('`account`, `join`')->from(TABLE_TEAM)->where('root')->eq($projectID)->andWhere('type')->eq($projectType)->fetchPairs(); foreach($accounts as $key => $account) { if(empty($account)) continue; if(!empty($project->days) and (int)$days[$key] > $project->days) { dao::$errors['message'][] = sprintf($this->lang->project->daysGreaterProject, $project->days); return false; } if((float)$hours[$key] > 24) { dao::$errors['message'][] = $this->lang->project->errorHours; return false; } } $this->dao->delete()->from(TABLE_TEAM)->where('root')->eq($projectID)->andWhere('type')->eq($projectType)->exec(); $projectMember = array(); foreach($accounts as $key => $account) { if(empty($account)) continue; $member = new stdclass(); $member->role = $roles[$key]; $member->days = $days[$key]; $member->hours = $hours[$key]; $member->limited = isset($limited[$key]) ? $limited[$key] : 'no'; $member->root = $projectID; $member->account = $account; $member->join = isset($oldJoin[$account]) ? $oldJoin[$account] : helper::today(); $member->type = $projectType; $projectMember[$account] = $member; $this->dao->insert(TABLE_TEAM)->data($member)->exec(); } /* Only changed account update userview. */ $oldAccounts = array_keys($oldJoin); $removedAccounts = array_diff($oldAccounts, $accounts); $changedAccounts = array_merge($removedAccounts, array_diff($accounts, $oldAccounts)); $changedAccounts = array_unique($changedAccounts); $childSprints = $this->dao->select('id')->from(TABLE_PROJECT)->where('project')->eq($projectID)->andWhere('type')->in('stage,sprint')->andWhere('deleted')->eq('0')->fetchPairs(); $linkedProducts = $this->dao->select("t2.id")->from(TABLE_PROJECTPRODUCT)->alias('t1') ->leftJoin(TABLE_PRODUCT)->alias('t2')->on('t1.product = t2.id') ->where('t2.deleted')->eq(0) ->andWhere('t1.project')->eq($projectID) ->andWhere('t2.vision')->eq($this->config->vision) ->fetchPairs(); $this->loadModel('user')->updateUserView(array($projectID), 'project', $changedAccounts); if(!empty($childSprints)) $this->user->updateUserView($childSprints, 'sprint', $changedAccounts); if(!empty($linkedProducts)) $this->user->updateUserView(array_keys($linkedProducts), 'product', $changedAccounts); /* Remove execution members. */ if($this->post->removeExecution == 'yes' and !empty($childSprints) and !empty($removedAccounts)) { $this->dao->delete()->from(TABLE_TEAM) ->where('root')->in($childSprints) ->andWhere('type')->eq('execution') ->andWhere('account')->in($removedAccounts) ->exec(); } if(empty($project->multiple) and $project->model != 'waterfall') $this->loadModel('execution')->syncNoMultipleSprint($projectID); } /** * Print datatable cell. * * @param object $col * @param object $project * @param array $users * @param int $programID * @access public * @return void */ public function printCell($col, $project, $users, $programID = 0) { $canOrder = common::hasPriv('project', 'updateOrder'); $canBatchEdit = common::hasPriv('project', 'batchEdit'); $account = $this->app->user->account; $id = $col->id; $projectLink = helper::createLink('project', 'index', "projectID=$project->id", '', '', $project->id); if($col->show) { $title = ''; $class = "c-$id" . (in_array($id, array('budget', 'teamCount', 'estimate', 'consume')) ? ' c-number' : ''); if($id == 'id') $class .= ' cell-id'; if($id == 'code') { $class .= ' c-name'; $title = "title={$project->code}"; } elseif($id == 'name') { $class .= ' text-left'; $title = "title='{$project->name}'"; } elseif($id == 'PM') { $class .= ' c-manager'; } if($id == 'end') { $project->end = $project->end == LONG_TIME ? $this->lang->project->longTime : $project->end; $class .= ' c-name'; $title = "title='{$project->end}'"; } if($id == 'budget') { $projectBudget = $this->getBudgetWithUnit($project->budget); $budgetTitle = $project->budget != 0 ? zget($this->lang->project->currencySymbol, $project->budgetUnit) . ' ' . $projectBudget : $this->lang->project->future; $title = "title='$budgetTitle'"; } if($id == 'estimate') $title = "title='{$project->hours->totalEstimate} {$this->lang->execution->workHour}'"; if($id == 'consume') $title = "title='{$project->hours->totalConsumed} {$this->lang->execution->workHour}'"; if($id == 'surplus') $title = "title='{$project->hours->totalLeft} {$this->lang->execution->workHour}'"; echo ""; if($this->config->edition != 'open') $this->loadModel('flow')->printFlowCell('project', $project, $id); switch($id) { case 'id': if($canBatchEdit) { echo html::checkbox('projectIdList', array($project->id => '')) . html::a($projectLink, sprintf('%03d', $project->id)); } else { printf('%03d', $project->id); } break; case 'name': $prefix = ''; $suffix = ''; $projectIcon = ''; if(isset($project->delay)) $suffix = "{$this->lang->project->statusList['delay']}"; $projectType = $project->model == 'scrum' ? 'sprint' : $project->model; if(!empty($suffix) or !empty($prefix)) echo '
    '; if(!empty($prefix)) echo $prefix; if($this->config->vision == 'rnd') $projectIcon = " "; echo html::a($projectLink, $projectIcon . $project->name, '', "class='text-ellipsis'"); if(!empty($suffix)) echo $suffix; if(!empty($suffix) or !empty($prefix)) echo '
    '; break; case 'code': echo $project->code; break; case 'PM': $user = $this->loadModel('user')->getByID($project->PM, 'account'); $userID = !empty($user) ? $user->id : ''; $userAvatar = !empty($user) ? $user->avatar : ''; $PMLink = helper::createLink('user', 'profile', "userID=$userID", '', true); $userName = zget($users, $project->PM); if($project->PM) echo html::smallAvatar(array('avatar' => $userAvatar, 'account' => $project->PM, 'name' => $userName), "avatar-circle avatar-{$project->PM}"); echo empty($project->PM) ? '' : html::a($PMLink, $userName, '', "title='{$userName}' data-toggle='modal' data-type='iframe' data-width='600'"); break; case 'begin': echo $project->begin; break; case 'end': echo $project->end; break; case 'status': echo " " . zget($this->lang->project->statusList, $project->status) . ""; break; case 'hasProduct': echo zget($this->lang->project->projectTypeList, $project->hasProduct); break; case 'budget': echo $budgetTitle; break; case 'teamCount': echo $project->teamCount; break; case 'estimate': echo $project->hours->totalEstimate . $this->lang->execution->workHourUnit; break; case 'consume': echo $project->hours->totalConsumed . $this->lang->execution->workHourUnit; break; case 'surplus': echo $project->hours->totalLeft . $this->lang->execution->workHourUnit; break; case 'progress': echo html::ring($project->hours->progress); break; case 'actions': $project->programID = $programID; echo $this->buildOperateMenu($project, 'browse'); break; } echo ''; } } /** * Convert budget unit. * * @param int $budget * @access public * @return void */ public function getBudgetWithUnit($budget) { if($budget < 10000) { $budget = round((float)$budget, 2); $unit = ''; } elseif($budget < 100000000 and $budget >= 10000) { $budget = round((float)$budget/10000, 2); $unit = $this->lang->project->tenThousand; } else { $budget = round((float)$budget/100000000, 2); $unit = $this->lang->project->hundredMillion; } $projectBudget = in_array($this->app->getClientLang(), array('zh-cn','zh-tw')) ? $budget . $unit : round((float)$budget, 2); return $projectBudget; } /** * Update products of a project. * * @param int $projectID * @param array $products * @access public * @return void */ public function updateProducts($projectID, $products = '') { $this->loadModel('user'); $members = array_keys($this->getTeamMembers($projectID)); /* Link products of other programs. */ if(!empty($_POST['otherProducts'])) { $products = array(); $otherProducts = $_POST['otherProducts']; foreach($otherProducts as $index => $otherProduct) { if(!$otherProduct) continue; $data = new stdclass(); $data->project = $projectID; $data->plan = 0; if(strpos($otherProduct, '_') !== false) { $params = explode('_', $otherProduct); $data->product = $params[0]; $data->branch = $params[1]; } else { $data->product = $otherProduct; $data->branch = 0; } $this->dao->insert(TABLE_PROJECTPRODUCT)->data($data)->exec(); $products[] = $data->product; } $this->user->updateUserView($products, 'product', $members); if((int)$projectID > 0 and !empty($_POST['division'])) { $this->dao->update(TABLE_PROJECT)->set('division')->eq('1')->where('id')->eq((int)$projectID)->exec(); $this->dao->update(TABLE_EXECUTION)->set('division')->eq('1')->where('project')->eq((int)$projectID)->exec(); } return !dao::isError(); } /* Link products of current program of the project. */ $products = isset($_POST['products']) ? $_POST['products'] : $products; $oldProjectProducts = $this->dao->select('*')->from(TABLE_PROJECTPRODUCT)->where('project')->eq((int)$projectID)->fetchGroup('product', 'branch'); $this->dao->delete()->from(TABLE_PROJECTPRODUCT)->where('project')->eq((int)$projectID)->exec(); if(empty($products)) { $this->user->updateUserView(array_keys($oldProjectProducts), 'product', $members); return true; } $branches = isset($_POST['branch']) ? $_POST['branch'] : array(); $plans = isset($_POST['plans']) ? $_POST['plans'] : array();; $existedProducts = array(); foreach($products as $i => $productID) { if(empty($productID)) continue; if(!isset($existedProducts[$productID])) $existedProducts[$productID] = array(); $oldPlan = 0; $branch = isset($branches[$i]) ? $branches[$i] : 0; if(!is_array($branch)) $branch = array($branch => $branch); foreach($branch as $branchID) { if(isset($existedProducts[$productID][$branchID])) continue; if(isset($oldProjectProducts[$productID][$branchID])) { $oldProjectProduct = $oldProjectProducts[$productID][$branchID]; if($this->app->rawMethod != 'edit') $oldPlan = $oldProjectProduct->plan; } $data = new stdclass(); $data->project = $projectID; $data->product = $productID; $data->branch = $branchID; $data->plan = isset($plans[$productID]) ? implode(',', $plans[$productID]) : $oldPlan; $data->plan = trim($data->plan, ','); $data->plan = empty($data->plan) ? 0 : ",$data->plan,"; $this->dao->insert(TABLE_PROJECTPRODUCT)->data($data)->exec(); $existedProducts[$productID][$branchID] = true; } } /* Delete the execution linked products that is not linked with the execution. */ $projectID = (int)$projectID; if($projectID > 0) { $executions = $this->dao->select('id')->from(TABLE_EXECUTION)->where('project')->eq($projectID)->fetchPairs('id'); $this->dao->delete()->from(TABLE_PROJECTPRODUCT)->where('project')->in($executions)->andWhere('product')->notin($products)->exec(); if(!empty($_POST['division'])) { $this->dao->update(TABLE_PROJECT)->set('division')->eq('1')->where('id')->eq((int)$projectID)->exec(); $this->dao->update(TABLE_EXECUTION)->set('division')->eq('1')->where('project')->eq((int)$projectID)->exec(); } $project = $this->getByID($projectID); if(!empty($project) and ($project->model == 'waterfall' or $project->model == 'waterfallplus') and empty($project->division) and !empty($executions)) { $this->loadModel('execution'); foreach($executions as $executionID) $this->execution->updateProducts($executionID); } } $oldProductKeys = array_keys($oldProjectProducts); $needUpdate = array_merge(array_diff($oldProductKeys, $products), array_diff($products, $oldProductKeys)); if($needUpdate) $this->user->updateUserView($needUpdate, 'product', $members); } /** * Update userview for involved product and execution. * * @param int $projectID * @param array $users * @access public * @return void */ public function updateInvolvedUserView($projectID, $users = array()) { $products = $this->dao->select('product')->from(TABLE_PROJECTPRODUCT)->where('project')->eq($projectID)->fetchPairs('product', 'product'); $this->loadModel('user')->updateUserView($products, 'product', $users); $executions = $this->dao->select('id')->from(TABLE_EXECUTION)->where('project')->eq($projectID)->fetchPairs('id', 'id'); if($executions) $this->user->updateUserView($executions, 'sprint', $users); } /** * Get team members. * * @param int $projectID * @access public * @return array */ public function getTeamMembers($projectID) { if(defined('TUTORIAL')) return $this->loadModel('tutorial')->getTeamMembers(); $project = $this->getByID($projectID); if(empty($project)) return array(); return $this->dao->select("t1.*, t1.hours * t1.days AS totalHours, t2.id as userID, if(t2.deleted='0', t2.realname, t1.account) as realname")->from(TABLE_TEAM)->alias('t1') ->leftJoin(TABLE_USER)->alias('t2')->on('t1.account = t2.account') ->where('t1.root')->eq((int)$projectID) ->andWhere('t1.type')->eq($project->type) ->andWhere('t2.deleted')->eq('0') ->beginIF($this->config->vision)->andWhere("CONCAT(',', t2.visions, ',')")->like("%,{$this->config->vision},%")->fi() ->fetchAll('account'); } /** * Get team member pairs by projectID. * * @param int $projectID * @access public * @return array */ public function getTeamMemberPairs($projectID) { $project = $this->dao->select('*')->from(TABLE_PROJECT)->where('id')->eq($projectID)->fetch(); if(empty($project)) return array(); $type = 'project'; if($project->type == 'sprint' or $project->type == 'stage' or $project->type == 'kanban') $type = 'execution'; $members = $this->dao->select("t1.account, if(t2.deleted='0', t2.realname, t1.account) as realname")->from(TABLE_TEAM)->alias('t1') ->leftJoin(TABLE_USER)->alias('t2')->on('t1.account = t2.account') ->where('t1.root')->eq((int)$projectID) ->andWhere('t1.type')->eq($type) ->andWhere('t2.deleted')->eq('0') ->beginIF($this->config->vision)->andWhere("CONCAT(',', t2.visions, ',')")->like("%,{$this->config->vision},%")->fi() ->fetchPairs('account', 'realname'); return array('' => '') + $members; } /** * Get team member group. * * @param array|string $projectIdList * @access public * @return array */ public function getTeamMemberGroup($projectIdList) { if(empty($projectIdList)) return array(); return $this->dao->select("t1.account, if(t2.deleted='0', t2.realname, t1.account) as realname, t1.root as project")->from(TABLE_TEAM)->alias('t1') ->leftJoin(TABLE_USER)->alias('t2')->on('t1.account = t2.account') ->where('t1.root')->in($projectIdList) ->andWhere('t1.type')->eq('project') ->andWhere('t2.deleted')->eq('0') ->beginIF($this->config->vision)->andWhere("CONCAT(',', t2.visions, ',')")->like("%,{$this->config->vision},%")->fi() ->fetchGroup('project', 'account'); } /** * Get members of a project who can be imported. * * @param int $projectID * @param array $currentMembers * @access public * @return array */ public function getMembers2Import($projectID, $currentMembers) { if($projectID == 0) return array(); return $this->dao->select('account, role, hours') ->from(TABLE_TEAM) ->where('root')->eq($projectID) ->andWhere('type')->eq('project') ->andWhere('account')->notIN($currentMembers) ->fetchAll('account'); } /** * Get stats for project kanban. * * @access public * @return void */ public function getStats4Kanban() { $this->loadModel('program'); $projects = $this->program->getProjectStats(0, 'all', 0, 'order_asc'); $executions = $this->loadModel('execution')->getStatData(0, 'doing', 0, 0, false, 'hasParentName|skipParent'); $doingExecutions = array(); $latestExecutions = array(); foreach($executions as $execution) $doingExecutions[$execution->project][$execution->id] = $execution; foreach($doingExecutions as $projectID => $executions) { krsort($doingExecutions[$projectID]); $latestExecutions[$projectID] = current($doingExecutions[$projectID]); } $myProjects = array(); $otherProjects = array(); $closedGroup = array(); foreach($projects as $project) { if(strpos('wait,doing,closed', $project->status) === false) continue; $projectPath = explode(',', trim($project->path, ',')); $topProgram = !empty($project->parent) ? $projectPath[0] : $project->parent; if($project->PM == $this->app->user->account) { if($project->status != 'closed') { $myProjects[$topProgram][$project->status][] = $project; } else { $closedGroup['my'][$topProgram][$project->closedDate] = $project; } } else { if($project->status != 'closed') { $otherProjects[$topProgram][$project->status][] = $project; } else { $closedGroup['other'][$topProgram][$project->closedDate] = $project; } } } /* Only display recent two closed projects. */ foreach($closedGroup as $group => $closedProjects) { foreach($closedProjects as $topProgram => $projects) { krsort($projects); if($group == 'my') { $myProjects[$topProgram]['closed'] = array_slice($projects, 0, 2); } else { $otherProjects[$topProgram]['closed'] = array_slice($projects, 0, 2); } } } return array('kanbanGroup' => array('my' => $myProjects, 'other' => $otherProjects), 'latestExecutions' => $latestExecutions); } /** * Computer execution progress. * * @param array $executions * @access public * @return array */ public function computerProgress($executions) { $hours = array(); $emptyHour = array('totalEstimate' => 0, 'totalConsumed' => 0, 'totalLeft' => 0, 'progress' => 0); /* Get all tasks and compute totalEstimate, totalConsumed, totalLeft, progress according to them. */ $tasks = $this->dao->select('id, execution, estimate, consumed, `left`, status, closedReason') ->from(TABLE_TASK) ->where('execution')->in(array_keys($executions)) ->andWhere('parent')->lt(1) ->andWhere('deleted')->eq(0) ->fetchGroup('execution', 'id'); $projects = $this->dao->select('t1.id,t2.model') ->from(TABLE_PROJECT)->alias('t1') ->leftJoin(TABLE_PROJECT)->alias('t2')->on('t1.project=t2.id') ->where('t1.deleted')->eq('0') ->andWhere('t1.id')->in(array_keys($executions)) ->fetchPairs(); /* Compute totalEstimate, totalConsumed, totalLeft. */ foreach($tasks as $executionID => $executionTasks) { $hour = (object)$emptyHour; foreach($executionTasks as $task) { $hour->totalEstimate += $task->estimate; $hour->totalConsumed += $task->consumed; if($task->status != 'cancel' and $task->status != 'closed') $hour->totalLeft += $task->left; } $hours[$executionID] = $hour; if(isset($executions[$executionID]) and $executions[$executionID]->grade > 1 and isset($projects[$executionID]) and ($projects[$executionID] == 'waterfall' or $projects[$executionID] == 'waterfallplus')) { $stageParents = $this->dao->select('id')->from(TABLE_EXECUTION)->where('id')->in(trim($executions[$executionID]->path, ','))->andWhere('type')->eq('stage')->andWhere('id')->ne($executions[$executionID]->id)->orderBy('grade')->fetchPairs(); foreach($stageParents as $stageParent) { if(!isset($hours[$stageParent])) { $hours[$stageParent] = clone $hour; continue; } $hours[$stageParent]->totalEstimate += $hour->totalEstimate; $hours[$stageParent]->totalConsumed += $hour->totalConsumed; $hours[$stageParent]->totalLeft += $hour->totalLeft; } } } /* Compute totalReal and progress. */ foreach($hours as $hour) { $hour->totalEstimate = round($hour->totalEstimate, 1) ; $hour->totalConsumed = round($hour->totalConsumed, 1); $hour->totalLeft = round($hour->totalLeft, 1); $hour->totalReal = $hour->totalConsumed + $hour->totalLeft; $hour->progress = $hour->totalReal ? round($hour->totalConsumed / $hour->totalReal * 100, 2) : 0; } return $hours; } /** * Set menu of project module. * * @param int $objectID projectID * @access public * @return int */ public function setMenu($objectID) { global $lang; $model = 'scrum'; $objectID = (empty($objectID) and $this->session->project) ? $this->session->project : $objectID; $project = $this->getByID($objectID); if(!$project) { $execution = $this->loadModel('execution')->getByID($objectID); if($execution and $execution->project and !$execution->multiple) { $project = $this->getByID($execution->project); $objectID = $execution->project; } } if($project and $project->model == 'waterfall') $model = $project->model; if($project and $project->model == 'waterfallplus') $model = 'waterfall'; if($project and $project->model == 'kanban') { $model = $project->model . 'Project'; $lang->executionCommon = $lang->project->kanban; } if(isset($lang->$model)) { $lang->project->menu = $lang->{$model}->menu; $lang->project->menuOrder = $lang->{$model}->menuOrder; $lang->project->dividerMenu = $lang->{$model}->dividerMenu; } if(empty($project->hasProduct)) { unset($lang->project->menu->settings['subMenu']->products); $projectProduct = $this->dao->select('product')->from(TABLE_PROJECTPRODUCT)->where('project')->eq($objectID)->fetch('product'); $lang->project->menu->settings['subMenu']->module['link'] = sprintf($lang->project->menu->settings['subMenu']->module['link'], $projectProduct); if(isset($project->model) and ($project->model == 'scrum' or $project->model == 'agileplus')) { $lang->project->menu->projectplan['link'] = sprintf($lang->project->menu->projectplan['link'], $projectProduct); } else { unset($lang->project->menu->projectplan); } if(!empty($this->config->URAndSR) && $project->model !== 'kanban' && isset($lang->project->menu->storyGroup)) { $lang->project->menu->settings['subMenu']->module['link'] = sprintf($lang->project->menu->settings['subMenu']->module['link'], $projectProduct); if($project->model !== 'kanban' && isset($lang->project->menu->storyGroup)) { $lang->project->menu->story = $lang->project->menu->storyGroup; $lang->project->menu->story['link'] = sprintf($lang->project->menu->storyGroup['link'], '%s', $projectProduct); $lang->project->menu->story['dropMenu']->story['link'] = sprintf($lang->project->menu->storyGroup['dropMenu']->story['link'], '%s', $projectProduct); $lang->project->menu->story['dropMenu']->requirement['link'] = sprintf($lang->project->menu->storyGroup['dropMenu']->requirement['link'], '%s', $projectProduct); } unset($lang->project->menu->settings['subMenu']->products); } } else { unset($lang->project->menu->settings['subMenu']->module); unset($lang->project->menu->settings['subMenu']->managerepo); unset($lang->project->menu->projectplan); } if(isset($lang->project->menu->storyGroup)) unset($lang->project->menu->storyGroup); /* Reset project priv. */ $moduleName = $this->app->rawModule; $methodName = $this->app->rawMethod; $this->loadModel('common')->resetProjectPriv($objectID); if(!$this->common->isOpenMethod($moduleName, $methodName) and !commonModel::hasPriv($moduleName, $methodName)) $this->common->deny($moduleName, $methodName, false); if(isset($project->model) and ($project->model == 'waterfall' or $project->model == 'waterfallplus')) { $lang->project->createExecution = str_replace($lang->executionCommon, $lang->project->stage, $lang->project->createExecution); $lang->project->lastIteration = str_replace($lang->executionCommon, $lang->project->stage, $lang->project->lastIteration); $this->loadModel('execution'); $executionCommonLang = $lang->executionCommon; $lang->executionCommon = $lang->project->stage; include $this->app->getModulePath('', 'execution') . 'lang/' . $this->app->getClientLang() . '.php'; $lang->execution->typeList['sprint'] = $executionCommonLang; } $lang->switcherMenu = $this->getSwitcher($objectID, $moduleName, $methodName); $this->saveState($objectID, $this->getPairsByProgram()); if(isset($project->acl) and $project->acl == 'open') unset($lang->project->menu->settings['subMenu']->whitelist); common::setMenuVars('project', $objectID); $this->setNoMultipleMenu($objectID); return $objectID; } /** * Set multi-scrum menu. * * @param int $objectID * @access public * @return void */ public function setNoMultipleMenu($objectID) { $moduleName = $this->app->rawModule; $methodName = $this->app->rawMethod; $this->session->set('multiple', true); $project = $this->dao->select('*')->from(TABLE_PROJECT)->where('id')->eq($objectID)->andWhere('multiple')->eq('0')->fetch(); if(empty($project)) return; if(!in_array($project->type, array('project', 'sprint', 'kanban'))) return; if($project->type == 'project') { $model = $project->model; $projectID = $project->id; $executionID = $this->dao->select('id')->from(TABLE_EXECUTION) ->where('project')->eq($projectID) ->andWhere('multiple')->eq('0') ->andWhere('type')->in('kanban,sprint') ->andWhere('deleted')->eq('0') ->fetch('id'); } else { $model = $project->type == 'kanban' ? 'kanban' : 'scrum'; $executionID = $project->id; $projectID = $project->project; } if(empty($projectID) or empty($executionID)) return; $this->session->set('project', $projectID, 'project'); $this->session->set('multiple', false); $navGroup = zget($this->lang->navGroup, $moduleName); $this->lang->$navGroup->menu = $this->lang->project->noMultiple->{$model}->menu; $this->lang->$navGroup->menuOrder = $this->lang->project->noMultiple->{$model}->menuOrder; $this->lang->$navGroup->dividerMenu = $this->lang->project->noMultiple->{$model}->dividerMenu; /* Single execution and has no product project menu. */ if(!$project->hasProduct and !$project->multiple and !empty($this->config->URAndSR)) { if(isset($this->lang->$navGroup->menu->storyGroup)) { $this->lang->$navGroup->menu->story = $this->lang->$navGroup->menu->storyGroup; $this->lang->$navGroup->menu->story['link'] = sprintf($this->lang->$navGroup->menu->storyGroup['link'], '%s', $projectID); $this->lang->$navGroup->menu->story['dropMenu']->story['link'] = sprintf($this->lang->$navGroup->menu->storyGroup['dropMenu']->story['link'], '%s', $projectID); $this->lang->$navGroup->menu->story['dropMenu']->requirement['link'] = sprintf($this->lang->$navGroup->menu->storyGroup['dropMenu']->requirement['link'], '%s', $projectID); } } if(isset($this->lang->$navGroup->menu->storyGroup)) unset($this->lang->$navGroup->menu->storyGroup); foreach($this->lang->$navGroup->menu as $label => $menu) { $objectID = 0; if(strpos($this->config->project->multiple['project'], ",{$label},") !== false) $objectID = $projectID; if(strpos($this->config->project->multiple['execution'], ",{$label},") !== false) { $objectID = $executionID; $this->lang->$navGroup->menu->{$label}['subModule'] = 'project'; } $this->lang->$navGroup->menu->$label = commonModel::setMenuVarsEx($menu, $objectID); if(isset($menu['subMenu'])) { foreach($menu['subMenu'] as $key1 => $subMenu) $this->lang->$navGroup->menu->{$label}['subMenu']->$key1 = common::setMenuVarsEx($subMenu, $objectID); } if(!isset($menu['dropMenu'])) continue; foreach($menu['dropMenu'] as $key2 => $dropMenu) { $this->lang->$navGroup->menu->{$label}['dropMenu']->$key2 = common::setMenuVarsEx($dropMenu, $objectID); if(!isset($dropMenu['subMenu'])) continue; foreach($dropMenu['subMenu'] as $key3 => $subMenu) $this->lang->$navGroup->menu->{$label}['dropMenu']->$key3 = common::setMenuVarsEx($subMenu, $objectID); } } /* If objectID is set, cannot use homeMenu. */ unset($this->lang->project->homeMenu); $this->lang->switcherMenu = $this->getSwitcher($projectID, $moduleName, $methodName); $this->lang->project->menu = $this->lang->$navGroup->menu; $this->lang->project->menuOrder = $this->lang->$navGroup->menuOrder; $this->lang->project->dividerMenu = $this->lang->$navGroup->dividerMenu; if(empty($project->hasProduct)) { unset($this->lang->project->menu->settings['subMenu']->products); } else { unset($this->lang->project->menu->settings['subMenu']->module); unset($this->lang->project->menu->settings['subMenu']->managerepo); unset($this->lang->project->menu->projectplan); } $this->loadModel('common')->resetProjectPriv($projectID); } /** * Check if the project model can be changed. * * @param int $projectID * @param string $model * @access public * @return bool */ public function checkCanChangeModel($projectID, $model) { $checkList = $this->config->project->checkList->$model; if($this->config->edition == 'max') $checkList = $this->config->project->maxCheckList->$model; foreach($checkList as $module) { if($module == '') continue; $type = ''; $table = constant('TABLE_'. strtoupper($module)); if($module == 'execution') { switch($model) { case 'scrum': $type = 'sprint'; break; case 'waterfall': $type = 'stage'; break; case 'kanban': $type = 'kanban'; break; } } $object = $this->getDataByProject($table, $projectID, $type); if(!empty($object)) return false; } return true; } /** * Get the objects under the project. * * @param constant $table * @param int $projectID * @param string $type * @access public * @return object */ public function getDataByProject($table, $projectID, $type = '') { $result = $this->dao->select('id')->from($table) ->where('project')->eq($projectID) ->beginIF(!empty($type))->andWhere('type')->eq($type)->fi() ->fetch(); return $result; } /** * Build project action menu. * * @param object $project * @param string $type * @access public * @return string */ public function buildOperateMenu($project, $type = 'view') { $function = 'buildOperate' . ucfirst($type) . 'Menu'; return $this->$function($project); } /** * Build project view action menu. * * @param object $project * @access public * @return string */ public function buildOperateViewMenu($project) { if($project->deleted) return ''; $menu = ''; $params = "projectID=$project->id"; $menu .= "
    "; $menu .= $this->buildMenu('project', 'start', $params, $project, 'view', 'play', '', 'iframe', true, '', $this->lang->project->start); $menu .= $this->buildMenu('project', 'activate', $params, $project, 'view', 'magic', '', 'iframe', true, '', $this->lang->project->activate); $menu .= $this->buildMenu('project', 'suspend', $params, $project, 'view', 'pause', '', 'iframe', true, '', $this->lang->project->suspend); $menu .= $this->buildMenu('project', 'close', $params, $project, 'view', 'off', '', 'iframe', true, '', $this->lang->close); $menu .= "
    "; $menu .= $this->buildFlowMenu('project', $project, 'view', 'direct'); $menu .= "
    "; $menu .= $this->buildMenu('project', 'edit', "project=$project->id&from=view", $project, 'button', 'edit', '', '', '', '', $this->lang->edit); $menu .= $this->buildMenu('project', 'delete', "project=$project->id&confirm=no&from=view", $project, 'button', 'trash', 'hiddenwin', '', '', '', $this->lang->delete); return $menu; } /** * Build project browse action menu. * * @param object $project * @access public * @return string */ public function buildOperateBrowseMenu($project) { $menu = ''; $params = "projectID=$project->id"; $moduleName = "project"; if($project->status == 'wait' || $project->status == 'suspended') { $menu .= $this->buildMenu($moduleName, 'start', $params, $project, 'browse', 'play', '', 'iframe', true); } if($project->status == 'doing') $menu .= $this->buildMenu($moduleName, 'close', $params, $project, 'browse', 'off', '', 'iframe', true); if($project->status == 'closed') $menu .= $this->buildMenu($moduleName, 'activate', $params, $project, 'browse', 'magic', '', 'iframe', true); if(common::hasPriv($moduleName, 'suspend') || (common::hasPriv($moduleName, 'close') && $project->status != 'doing') || (common::hasPriv($moduleName, 'activate') && $project->status != 'closed')) { $menu .= "
    "; $menu .= ""; $menu .= ""; $menu .= "
    "; } $from = $project->from == 'project' ? 'project' : 'pgmproject'; $iframe = $this->app->tab == 'program' ? 'iframe' : ''; $onlyBody = $this->app->tab == 'program' ? true : ''; $dataApp = "data-app=project"; $menu .= $this->buildMenu($moduleName, 'edit', $params, $project, 'browse', 'edit', '', $iframe, $onlyBody, $dataApp); if($this->config->vision != 'lite') { $menu .= $this->buildMenu($moduleName, 'team', $params, $project, 'browse', 'group', '', '', '', $dataApp, $this->lang->execution->team); $menu .= $this->buildMenu('project', 'group', "$params&programID={$project->programID}", $project, 'browse', 'lock', '', '', '', $dataApp); if(common::hasPriv($moduleName, 'manageProducts') || common::hasPriv($moduleName, 'whitelist') || common::hasPriv($moduleName, 'delete')) { $menu .= "
    "; $menu .= ""; $menu .= ""; $menu .= "
    "; } } else { $menu .= $this->buildMenu($moduleName, 'team', $params, $project, 'browse', 'group', '', '', '', $dataApp, $this->lang->execution->team); $menu .= $this->buildMenu('project', 'whitelist', "$params&module=project&from=$from", $project, 'browse', 'shield-check', '', 'btn-action', '', $dataApp); $menu .= $this->buildMenu($moduleName, "delete", $params, $project, 'browse', 'trash', 'hiddenwin', 'btn-action'); } return $menu; } /** * Update linked repos. This method only for project without product. * * @param int $projectID * @param array $repos * @access public * @return void */ public function updateRepoRelations($projectID, $repos) { $this->loadModel('action'); $this->loadModel('product'); $linkedRepos = $this->dao->select('*')->from(TABLE_REPO) ->where('deleted')->eq(0) ->andWhere("CONCAT(',', projects, ',')")->like("%,$projectID,%") ->fetchAll('id'); /* 1. Delete old relations. */ /* Get relations should be deleted. */ $shouldDeleteRepos = array_diff(array_keys($linkedRepos), $repos); foreach($shouldDeleteRepos as $repoID) { /* Remove project id form this repo. */ $repo = zget($linkedRepos, $repoID, null); $oldProjects = explode(',', $repo->projects); $newProjects = array_filter($oldProjects, function($oldProjectID) use ($projectID) { return $projectID != $oldProjectID; }); /* Remove products */ $shadowProduct = $this->loadModel('product')->getShadowProductByProject($projectID); $oldProducts = explode(',', $repo->product); $newProducts = array_filter($oldProducts, function($oldProductID)use($shadowProduct) { return $oldProductID != $shadowProduct->id; }); $this->dao->update(TABLE_REPO) ->set('product')->eq(implode(',', $newProducts)) ->set('projects')->eq(implode(',', $newProjects)) ->where('id')->eq($repo->id)->exec(); /* Save action log. */ $this->action->create('project', $projectID, 'unlinkedRepo', $repo->name); } /* 2. Add new relations. */ $products = $this->product->getProductPairsByProject($projectID); // There will be only one (shadow) product for project without product. $addedProducts = array_keys($products); $addedRepos = array_diff($repos, array_keys($linkedRepos)); $newRepos = $this->dao->select('*')->from(TABLE_REPO)->where('id')->in($addedRepos)->fetchAll(); foreach($newRepos as $repo) { $oldProducts = explode(',', $repo->product); $newProducts = array_merge($oldProducts, $addedProducts); $newProducts = array_unique($newProducts); $oldProjects = explode(',', $repo->projects); $oldProjects[] = $projectID; $newProjects = array_unique($oldProjects); $this->dao->update(TABLE_REPO) ->set('product')->eq(implode(',', $newProducts)) ->set('projects')->eq(implode(',', $newProjects)) ->where('id')->eq($repo->id)->exec(); $this->action->create('project', $projectID, 'linkedRepo', '', $repo->name); } } /** * Get linked repo pairs by project id. * * @param int $projectID * @access public * @return array */ public function linkedRepoPairs($projectID) { $repos = $this->dao->select('*')->from(TABLE_REPO) ->where('deleted')->eq(0) ->andWhere("CONCAT(',', projects, ',')")->like("%,$projectID,%") ->fetchAll(); $repoPairs = array(); foreach($repos as $repo) { $scm = $repo->SCM == 'Subversion' ? 'svn' : strtolower($repo->SCM); $repoPairs[$repo->id] = "[{$scm}] " . $repo->name; } return $repoPairs; } }