t('Yellow'), 'blue' => t('Blue'), 'green' => t('Green'), 'purple' => t('Purple'), 'red' => t('Red'), 'orange' => t('Orange'), 'grey' => t('Grey'), ); } /** * Get a list of due tasks for all projects * * @access public * @return array */ public function getOverdueTasks() { $tasks = $this->db->table(self::TABLE) ->columns( self::TABLE.'.id', self::TABLE.'.title', self::TABLE.'.date_due', self::TABLE.'.project_id', Project::TABLE.'.name AS project_name', User::TABLE.'.username AS assignee_username', User::TABLE.'.name AS assignee_name' ) ->join(Project::TABLE, 'id', 'project_id') ->join(User::TABLE, 'id', 'owner_id') ->eq(Project::TABLE.'.is_active', 1) ->eq(self::TABLE.'.is_active', 1) ->neq(self::TABLE.'.date_due', 0) ->lte(self::TABLE.'.date_due', mktime(23, 59, 59)) ->findAll(); return $tasks; } /** * Fetch one task * * @access public * @param integer $task_id Task id * @param boolean $more If true, fetch all related information * @return array */ public function getById($task_id, $more = false) { if ($more) { $sql = ' SELECT tasks.id, tasks.title, tasks.description, tasks.date_creation, tasks.date_completed, tasks.date_modification, tasks.date_due, tasks.color_id, tasks.project_id, tasks.column_id, tasks.owner_id, tasks.creator_id, tasks.position, tasks.is_active, tasks.score, tasks.category_id, project_has_categories.name AS category_name, projects.name AS project_name, columns.title AS column_title, users.username AS assignee_username, users.name AS assignee_name, creators.username AS creator_username, creators.name AS creator_name FROM tasks LEFT JOIN users ON users.id = tasks.owner_id LEFT JOIN users AS creators ON creators.id = tasks.creator_id LEFT JOIN project_has_categories ON project_has_categories.id = tasks.category_id LEFT JOIN projects ON projects.id = tasks.project_id LEFT JOIN columns ON columns.id = tasks.column_id WHERE tasks.id = ? '; $rq = $this->db->execute($sql, array($task_id)); return $rq->fetch(PDO::FETCH_ASSOC); } else { return $this->db->table(self::TABLE)->eq('id', $task_id)->findOne(); } } /** * Count all tasks for a given project and status * * @access public * @param integer $project_id Project id * @param integer $status_id Status id * @return array */ public function getAll($project_id, $status_id = self::STATUS_OPEN) { return $this->db ->table(self::TABLE) ->eq('project_id', $project_id) ->eq('is_active', $status_id) ->findAll(); } /** * Count all tasks for a given project and status * * @access public * @param integer $project_id Project id * @param array $status List of status id * @return integer */ public function countByProjectId($project_id, array $status = array(self::STATUS_OPEN, self::STATUS_CLOSED)) { return $this->db ->table(self::TABLE) ->eq('project_id', $project_id) ->in('is_active', $status) ->count(); } /** * Get tasks that match defined filters * * @access public * @param array $filters Filters: [ ['column' => '...', 'operator' => '...', 'value' => '...'], ... ] * @param array $sorting Sorting: [ 'column' => 'date_creation', 'direction' => 'asc'] * @return array */ public function find(array $filters, array $sorting = array()) { $table = $this->db ->table(self::TABLE) ->columns( '(SELECT count(*) FROM comments WHERE task_id=tasks.id) AS nb_comments', '(SELECT count(*) FROM task_has_files WHERE task_id=tasks.id) AS nb_files', '(SELECT count(*) FROM task_has_subtasks WHERE task_id=tasks.id) AS nb_subtasks', '(SELECT count(*) FROM task_has_subtasks WHERE task_id=tasks.id AND status=2) AS nb_completed_subtasks', 'tasks.id', 'tasks.title', 'tasks.description', 'tasks.date_creation', 'tasks.date_modification', 'tasks.date_completed', 'tasks.date_due', 'tasks.color_id', 'tasks.project_id', 'tasks.column_id', 'tasks.owner_id', 'tasks.creator_id', 'tasks.position', 'tasks.is_active', 'tasks.score', 'tasks.category_id', 'users.username AS assignee_username', 'users.name AS assignee_name' ) ->join(User::TABLE, 'id', 'owner_id'); foreach ($filters as $key => $filter) { if ($key === 'or') { $table->beginOr(); foreach ($filter as $subfilter) { $table->$subfilter['operator']($subfilter['column'], $subfilter['value']); } $table->closeOr(); } else if (isset($filter['operator']) && isset($filter['column']) && isset($filter['value'])) { $table->$filter['operator']($filter['column'], $filter['value']); } } if (empty($sorting)) { $table->orderBy('tasks.position', 'ASC'); } else { $table->orderBy($sorting['column'], $sorting['direction']); } return $table->findAll(); } /** * Count the number of tasks for a given column and status * * @access public * @param integer $project_id Project id * @param integer $column_id Column id * @param array $status List of status id * @return integer */ public function countByColumnId($project_id, $column_id, array $status = array(self::STATUS_OPEN)) { return $this->db ->table(self::TABLE) ->eq('project_id', $project_id) ->eq('column_id', $column_id) ->in('is_active', $status) ->count(); } /** * Generic method to duplicate a task * * @access public * @param array $task Task data * @param array $override Task properties to override * @return integer|boolean */ public function copy(array $task, array $override = array()) { // Values to override if (! empty($override)) { $task = $override + $task; } $this->db->startTransaction(); // Assign new values $values = array(); $values['title'] = $task['title']; $values['description'] = $task['description']; $values['date_creation'] = time(); $values['date_modification'] = $values['date_creation']; $values['date_due'] = $task['date_due']; $values['color_id'] = $task['color_id']; $values['project_id'] = $task['project_id']; $values['column_id'] = $task['column_id']; $values['owner_id'] = 0; $values['creator_id'] = $task['creator_id']; $values['position'] = $this->countByColumnId($values['project_id'], $values['column_id']) + 1; $values['score'] = $task['score']; $values['category_id'] = 0; // Check if the assigned user is allowed for the new project if ($task['owner_id'] && $this->project->isUserAllowed($values['project_id'], $task['owner_id'])) { $values['owner_id'] = $task['owner_id']; } // Check if the category exists if ($task['category_id'] && $this->category->exists($task['category_id'], $task['project_id'])) { $values['category_id'] = $task['category_id']; } // Save task if (! $this->db->table(self::TABLE)->save($values)) { $this->db->cancelTransaction(); return false; } $task_id = $this->db->getConnection()->getLastId(); // Duplicate subtasks if (! $this->subTask->duplicate($task['id'], $task_id)) { $this->db->cancelTransaction(); return false; } $this->db->closeTransaction(); // Trigger events $this->event->trigger(self::EVENT_CREATE_UPDATE, array('task_id' => $task_id) + $values); $this->event->trigger(self::EVENT_CREATE, array('task_id' => $task_id) + $values); return $task_id; } /** * Duplicate a task to the same project * * @access public * @param array $task Task data * @return integer|boolean */ public function duplicateSameProject($task) { return $this->copy($task); } /** * Duplicate a task to another project (always copy to the first column) * * @access public * @param integer $project_id Destination project id * @param array $task Task data * @return integer|boolean */ public function duplicateToAnotherProject($project_id, array $task) { return $this->copy($task, array( 'project_id' => $project_id, 'column_id' => $this->board->getFirstColumn($project_id), )); } /** * Prepare data before task creation or modification * * @access public * @param array $values Form values */ public function prepare(array &$values) { if (isset($values['another_task'])) { unset($values['another_task']); } if (! empty($values['date_due']) && ! is_numeric($values['date_due'])) { $values['date_due'] = $this->parseDate($values['date_due']); } // Force integer fields at 0 (for Postgresql) if (isset($values['date_due']) && empty($values['date_due'])) { $values['date_due'] = 0; } if (isset($values['score']) && empty($values['score'])) { $values['score'] = 0; } if (isset($values['is_active'])) { $values['is_active'] = (int) $values['is_active']; } } /** * Create a task * * @access public * @param array $values Form values * @return boolean */ public function create(array $values) { $this->db->startTransaction(); // Prepare data $this->prepare($values); if (empty($values['column_id'])) { $values['column_id'] = $this->board->getFirstColumn($values['project_id']); } if (empty($values['color_id'])) { $colors = $this->getColors(); $values['color_id'] = key($colors); } $values['date_creation'] = time(); $values['date_modification'] = $values['date_creation']; $values['position'] = $this->countByColumnId($values['project_id'], $values['column_id']) + 1; // Save task if (! $this->db->table(self::TABLE)->save($values)) { $this->db->cancelTransaction(); return false; } $task_id = $this->db->getConnection()->getLastId(); $this->db->closeTransaction(); // Trigger events $this->event->trigger(self::EVENT_CREATE_UPDATE, array('task_id' => $task_id) + $values); $this->event->trigger(self::EVENT_CREATE, array('task_id' => $task_id) + $values); return $task_id; } /** * Update a task * * @access public * @param array $values Form values * @param boolean $trigger_Events Trigger events * @return boolean */ public function update(array $values, $trigger_events = true) { // Fetch original task $original_task = $this->getById($values['id']); if (! $original_task) { return false; } // Prepare data $this->prepare($values); $updated_task = $values; $updated_task['date_modification'] = time(); unset($updated_task['id']); $result = $this->db->table(self::TABLE)->eq('id', $values['id'])->update($updated_task); if ($result && $trigger_events) { $this->triggerUpdateEvents($original_task, $updated_task); } return true; } /** * Trigger events for task modification * * @access public * @param array $original_task Original task data * @param array $updated_task Updated task data */ public function triggerUpdateEvents(array $original_task, array $updated_task) { $events = array(); if (isset($updated_task['owner_id']) && $original_task['owner_id'] != $updated_task['owner_id']) { $events[] = self::EVENT_ASSIGNEE_CHANGE; } else if (isset($updated_task['column_id']) && $original_task['column_id'] != $updated_task['column_id']) { $events[] = self::EVENT_MOVE_COLUMN; } else if (isset($updated_task['position']) && $original_task['position'] != $updated_task['position']) { $events[] = self::EVENT_MOVE_POSITION; } else { $events[] = self::EVENT_CREATE_UPDATE; $events[] = self::EVENT_UPDATE; } $event_data = array_merge($original_task, $updated_task); $event_data['task_id'] = $original_task['id']; foreach ($events as $event) { $this->event->trigger($event, $event_data); } } /** * Return true if the project exists * * @access public * @param integer $task_id Task id * @return boolean */ public function exists($task_id) { return $this->db->table(self::TABLE)->eq('id', $task_id)->count() === 1; } /** * Mark a task closed * * @access public * @param integer $task_id Task id * @return boolean */ public function close($task_id) { if (! $this->exists($task_id)) { return false; } $result = $this->db ->table(self::TABLE) ->eq('id', $task_id) ->update(array( 'is_active' => 0, 'date_completed' => time() )); if ($result) { $this->event->trigger(self::EVENT_CLOSE, array('task_id' => $task_id) + $this->getById($task_id)); } return $result; } /** * Mark a task open * * @access public * @param integer $task_id Task id * @return boolean */ public function open($task_id) { if (! $this->exists($task_id)) { return false; } $result = $this->db ->table(self::TABLE) ->eq('id', $task_id) ->update(array( 'is_active' => 1, 'date_completed' => 0 )); if ($result) { $this->event->trigger(self::EVENT_OPEN, array('task_id' => $task_id) + $this->getById($task_id)); } return $result; } /** * Remove a task * * @access public * @param integer $task_id Task id * @return boolean */ public function remove($task_id) { if (! $this->exists($task_id)) { return false; } $this->file->removeAll($task_id); return $this->db->table(self::TABLE)->eq('id', $task_id)->remove(); } /** * Move a task to another column or to another position * * @access public * @param integer $project_id Project id * @param integer $task_id Task id * @param integer $column_id Column id * @param integer $position Position (must be >= 1) * @return boolean */ public function movePosition($project_id, $task_id, $column_id, $position) { // The position can't be lower than 1 if ($position < 1) { return false; } $board = $this->db->table(Board::TABLE)->eq('project_id', $project_id)->asc('position')->findAllByColumn('id'); $columns = array(); // Prepare the columns foreach ($board as $board_column_id) { $columns[$board_column_id] = $this->db->table(self::TABLE) ->eq('is_active', 1) ->eq('project_id', $project_id) ->eq('column_id', $board_column_id) ->neq('id', $task_id) ->asc('position') ->findAllByColumn('id'); } // The column must exists if (! isset($columns[$column_id])) { return false; } // We put our task to the new position array_splice($columns[$column_id], $position - 1, 0, $task_id); // print_r($columns); // We save the new positions for all tasks return $this->savePositions($task_id, $columns); } /** * Save task positions * * @access private * @param integer $moved_task_id Id of the moved task * @param array $columns Sorted tasks * @return boolean */ private function savePositions($moved_task_id, array $columns) { $this->db->startTransaction(); foreach ($columns as $column_id => $column) { $position = 1; foreach ($column as $task_id) { if ($task_id == $moved_task_id) { // Events will be triggered only for that task $result = $this->update(array( 'id' => $task_id, 'position' => $position, 'column_id' => $column_id )); } else { $result = $this->db->table(self::TABLE)->eq('id', $task_id)->update(array( 'position' => $position, 'column_id' => $column_id )); } $position++; if (! $result) { $this->db->cancelTransaction(); return false; } } } $this->db->closeTransaction(); return true; } /** * Move a task to another project * * @access public * @param integer $project_id Project id * @param array $task Task data * @return boolean */ public function moveToAnotherProject($project_id, array $task) { $values = array(); // Clear values (categories are different for each project) $values['category_id'] = 0; $values['owner_id'] = 0; // Check if the assigned user is allowed for the new project if ($task['owner_id'] && $this->project->isUserAllowed($project_id, $task['owner_id'])) { $values['owner_id'] = $task['owner_id']; } // We use the first column of the new project $values['column_id'] = $this->board->getFirstColumn($project_id); $values['position'] = $this->countByColumnId($project_id, $values['column_id']) + 1; $values['project_id'] = $project_id; if ($this->db->table(self::TABLE)->eq('id', $task['id'])->update($values)) { return $task['id']; } return false; } /** * Common validation rules * * @access private * @return array */ private function commonValidationRules() { return array( new Validators\Integer('id', t('This value must be an integer')), new Validators\Integer('project_id', t('This value must be an integer')), new Validators\Integer('column_id', t('This value must be an integer')), new Validators\Integer('owner_id', t('This value must be an integer')), new Validators\Integer('creator_id', t('This value must be an integer')), new Validators\Integer('score', t('This value must be an integer')), new Validators\Integer('category_id', t('This value must be an integer')), new Validators\MaxLength('title', t('The maximum length is %d characters', 200), 200), new Validators\Date('date_due', t('Invalid date'), $this->getDateFormats()), ); } /** * Validate task creation * * @access public * @param array $values Form values * @return array $valid, $errors [0] = Success or not, [1] = List of errors */ public function validateCreation(array $values) { $rules = array( new Validators\Required('project_id', t('The project is required')), new Validators\Required('title', t('The title is required')), ); $v = new Validator($values, array_merge($rules, $this->commonValidationRules())); return array( $v->execute(), $v->getErrors() ); } /** * Validate description creation * * @access public * @param array $values Form values * @return array $valid, $errors [0] = Success or not, [1] = List of errors */ public function validateDescriptionCreation(array $values) { $rules = array( new Validators\Required('id', t('The id is required')), new Validators\Required('description', t('The description is required')), ); $v = new Validator($values, array_merge($rules, $this->commonValidationRules())); return array( $v->execute(), $v->getErrors() ); } /** * Validate task modification * * @access public * @param array $values Form values * @return array $valid, $errors [0] = Success or not, [1] = List of errors */ public function validateModification(array $values) { $rules = array( new Validators\Required('id', t('The id is required')), ); $v = new Validator($values, array_merge($rules, $this->commonValidationRules())); return array( $v->execute(), $v->getErrors() ); } /** * Validate assignee change * * @access public * @param array $values Form values * @return array $valid, $errors [0] = Success or not, [1] = List of errors */ public function validateAssigneeModification(array $values) { $rules = array( new Validators\Required('id', t('The id is required')), new Validators\Required('project_id', t('The project is required')), new Validators\Required('owner_id', t('This value is required')), ); $v = new Validator($values, array_merge($rules, $this->commonValidationRules())); return array( $v->execute(), $v->getErrors() ); } /** * Validate category change * * @access public * @param array $values Form values * @return array $valid, $errors [0] = Success or not, [1] = List of errors */ public function validateCategoryModification(array $values) { $rules = array( new Validators\Required('id', t('The id is required')), new Validators\Required('project_id', t('The project is required')), new Validators\Required('category_id', t('This value is required')), ); $v = new Validator($values, array_merge($rules, $this->commonValidationRules())); return array( $v->execute(), $v->getErrors() ); } /** * Validate project modification * * @access public * @param array $values Form values * @return array $valid, $errors [0] = Success or not, [1] = List of errors */ public function validateProjectModification(array $values) { $rules = array( new Validators\Required('id', t('The id is required')), new Validators\Required('project_id', t('The project is required')), ); $v = new Validator($values, array_merge($rules, $this->commonValidationRules())); return array( $v->execute(), $v->getErrors() ); } /** * Return a timestamp if the given date format is correct otherwise return 0 * * @access public * @param string $value Date to parse * @param string $format Date format * @return integer */ public function getValidDate($value, $format) { $date = DateTime::createFromFormat($format, $value); if ($date !== false) { $errors = DateTime::getLastErrors(); if ($errors['error_count'] === 0 && $errors['warning_count'] === 0) { $timestamp = $date->getTimestamp(); return $timestamp > 0 ? $timestamp : 0; } } return 0; } /** * Parse a date ad return a unix timestamp, try different date formats * * @access public * @param string $value Date to parse * @return integer */ public function parseDate($value) { foreach ($this->getDateFormats() as $format) { $timestamp = $this->getValidDate($value, $format); if ($timestamp !== 0) { return $timestamp; } } return null; } /** * Return the list of supported date formats * * @access public * @return array */ public function getDateFormats() { return array( t('m/d/Y'), 'Y-m-d', 'Y_m_d', ); } /** * For a given timestamp, reset the date to midnight * * @access public * @param integer $timestamp Timestamp * @return integer */ public function resetDateToMidnight($timestamp) { return mktime(0, 0, 0, date('m', $timestamp), date('d', $timestamp), date('Y', $timestamp)); } /** * Export a list of tasks for a given project and date range * * @access public * @param integer $project_id Project id * @param mixed $from Start date (timestamp or user formatted date) * @param mixed $to End date (timestamp or user formatted date) * @return array */ public function export($project_id, $from, $to) { $sql = ' SELECT tasks.id, projects.name AS project_name, tasks.is_active, project_has_categories.name AS category_name, columns.title AS column_title, tasks.position, tasks.color_id, tasks.date_due, creators.username AS creator_username, users.username AS assignee_username, tasks.score, tasks.title, tasks.date_creation, tasks.date_modification, tasks.date_completed FROM tasks LEFT JOIN users ON users.id = tasks.owner_id LEFT JOIN users AS creators ON creators.id = tasks.creator_id LEFT JOIN project_has_categories ON project_has_categories.id = tasks.category_id LEFT JOIN columns ON columns.id = tasks.column_id LEFT JOIN projects ON projects.id = tasks.project_id WHERE tasks.date_creation >= ? AND tasks.date_creation <= ? AND tasks.project_id = ? '; if (! is_numeric($from)) { $from = $this->resetDateToMidnight($this->parseDate($from)); } if (! is_numeric($to)) { $to = $this->resetDateToMidnight(strtotime('+1 day', $this->parseDate($to))); } $rq = $this->db->execute($sql, array($from, $to, $project_id)); $tasks = $rq->fetchAll(PDO::FETCH_ASSOC); $columns = array( e('Task Id'), e('Project'), e('Status'), e('Category'), e('Column'), e('Position'), e('Color'), e('Due date'), e('Creator'), e('Assignee'), e('Complexity'), e('Title'), e('Creation date'), e('Modification date'), e('Completion date'), ); $results = array($columns); foreach ($tasks as &$task) { $results[] = array_values($this->formatOutput($task)); } return $results; } /** * Format the output of a task array * * @access public * @param array $task Task properties * @return array */ public function formatOutput(array &$task) { $colors = $this->getColors(); $task['score'] = $task['score'] ?: ''; $task['is_active'] = $task['is_active'] == self::STATUS_OPEN ? e('Open') : e('Closed'); $task['color_id'] = $colors[$task['color_id']]; $task['date_creation'] = date('Y-m-d', $task['date_creation']); $task['date_due'] = $task['date_due'] ? date('Y-m-d', $task['date_due']) : ''; $task['date_modification'] = $task['date_modification'] ? date('Y-m-d', $task['date_modification']) : ''; $task['date_completed'] = $task['date_completed'] ? date('Y-m-d', $task['date_completed']) : ''; return $task; } }