18a638198SSzymon Olewniczak<?php 28a638198SSzymon Olewniczak 38a638198SSzymon Olewniczaknamespace dokuwiki\plugin\bez\mdl; 48a638198SSzymon Olewniczak 5*a5de966aSSzymon Olewniczakuse dokuwiki\plugin\bez\meta\ConsistencyViolationException; 68a638198SSzymon Olewniczakuse dokuwiki\plugin\bez\meta\Mailer; 78a638198SSzymon Olewniczakuse dokuwiki\plugin\bez\meta\PermissionDeniedException; 88a638198SSzymon Olewniczakuse dokuwiki\plugin\bez\meta\ValidationException; 914a1f0a4SSzymon Olewniczakuse dokuwiki\plugin\struct\types\DateTime; 108a638198SSzymon Olewniczak 118a638198SSzymon Olewniczakclass Task extends Entity { 128a638198SSzymon Olewniczak 138a638198SSzymon Olewniczak protected $id; 148a638198SSzymon Olewniczak 15e8827d73SSzymon Olewniczak protected $original_poster, $assignee, $closed_by; 168a638198SSzymon Olewniczak 178a638198SSzymon Olewniczak protected $private, $lock; 188a638198SSzymon Olewniczak 198a638198SSzymon Olewniczak protected $state, $type; 208a638198SSzymon Olewniczak 218a638198SSzymon Olewniczak protected $create_date, $last_activity_date, $last_modification_date, $close_date; 228a638198SSzymon Olewniczak 238a638198SSzymon Olewniczak protected $cost, $plan_date, $all_day_event, $start_time, $finish_time; 248a638198SSzymon Olewniczak 258a638198SSzymon Olewniczak protected $content, $content_html; 268a638198SSzymon Olewniczak 278a638198SSzymon Olewniczak protected $thread_id, $thread_comment_id, $task_program_id; 288a638198SSzymon Olewniczak 298a638198SSzymon Olewniczak /** @var \dokuwiki\plugin\bez\mdl\Thread */ 308a638198SSzymon Olewniczak protected $thread; 318a638198SSzymon Olewniczak 328a638198SSzymon Olewniczak /** @var Thread_comment */ 338a638198SSzymon Olewniczak protected $thread_comment; 348a638198SSzymon Olewniczak 358a638198SSzymon Olewniczak //virtual 36ff14b107SSzymon Olewniczak protected $task_program_name, $priority, $coordinator; 378a638198SSzymon Olewniczak 388a638198SSzymon Olewniczak public static function get_columns() { 398a638198SSzymon Olewniczak return array('id', 40e8827d73SSzymon Olewniczak 'original_poster', 'assignee', 'closed_by', 418a638198SSzymon Olewniczak 'private', 'lock', 428a638198SSzymon Olewniczak 'state', 'type', 438a638198SSzymon Olewniczak 'create_date', 'last_activity_date', 'last_modification_date', 'close_date', 448a638198SSzymon Olewniczak 'cost', 'plan_date', 'all_day_event', 'start_time', 'finish_time', 458a638198SSzymon Olewniczak 'content', 'content_html', 468a638198SSzymon Olewniczak 'thread_id', 'thread_comment_id', 'task_program_id'); 478a638198SSzymon Olewniczak } 488a638198SSzymon Olewniczak 49b331b892SSzymon Olewniczak public static function get_acl_columns() { 50b331b892SSzymon Olewniczak return array_merge(parent::get_acl_columns(), array('participants')); 51b331b892SSzymon Olewniczak } 52b331b892SSzymon Olewniczak 53e8827d73SSzymon Olewniczak public static function get_types() { 54e8827d73SSzymon Olewniczak return array('correction', 'corrective', 'preventive', 'program'); 55e8827d73SSzymon Olewniczak } 56e8827d73SSzymon Olewniczak 57e8827d73SSzymon Olewniczak public static function get_states() { 58e8827d73SSzymon Olewniczak return array('opened', 'done'); 59e8827d73SSzymon Olewniczak } 60e8827d73SSzymon Olewniczak 618a638198SSzymon Olewniczak public function __get($property) { 62ff14b107SSzymon Olewniczak if ($property == 'thread') { 63ff14b107SSzymon Olewniczak if ($this->thread_id == null) { 64ff14b107SSzymon Olewniczak return null; 65ff14b107SSzymon Olewniczak } 66ff14b107SSzymon Olewniczak if ($this->thread == null) { 67ff14b107SSzymon Olewniczak $this->thread = $this->model->threadFactory->get_one($this->thread_id); 68ff14b107SSzymon Olewniczak } 69ff14b107SSzymon Olewniczak return $this->thread; 70ff14b107SSzymon Olewniczak 71ff14b107SSzymon Olewniczak } elseif($property == 'thread_comment') { 72ff14b107SSzymon Olewniczak if ($this->thread_comment_id == null) { 73ff14b107SSzymon Olewniczak return null; 74ff14b107SSzymon Olewniczak } 75ff14b107SSzymon Olewniczak if ($this->thread_comment == null) { 7653df74e7SSzymon Olewniczak $this->thread_comment = $this->model->thread_commentFactory->get_one($this->thread_comment_id); 77ff14b107SSzymon Olewniczak } 78ff14b107SSzymon Olewniczak return $this->thread_comment; 79ff14b107SSzymon Olewniczak 80ff14b107SSzymon Olewniczak } elseif($property == 'priority' || $property == 'coordinator' || $property == 'task_program_name') { 818a638198SSzymon Olewniczak return $this->$property; 828a638198SSzymon Olewniczak } 838a638198SSzymon Olewniczak return parent::__get($property); 848a638198SSzymon Olewniczak } 858a638198SSzymon Olewniczak 868a638198SSzymon Olewniczak public function __construct($model, $defaults=array()) { 878a638198SSzymon Olewniczak parent::__construct($model, $defaults); 888a638198SSzymon Olewniczak 898a638198SSzymon Olewniczak $this->validator->set_rules(array( 908a638198SSzymon Olewniczak 'assignee' => array(array('dw_user'), 'NOT NULL'), 918a638198SSzymon Olewniczak 'cost' => array(array('numeric'), 'NULL'), 928a638198SSzymon Olewniczak 'plan_date' => array(array('iso_date'), 'NOT NULL'), 938a638198SSzymon Olewniczak 'all_day_event' => array(array('select', array('0', '1')), 'NOT NULL'), 948a638198SSzymon Olewniczak 'start_time' => array(array('time'), 'NULL'), 958a638198SSzymon Olewniczak 'finish_time' => array(array('time'), 'NULL'), 968a638198SSzymon Olewniczak 'content' => array(array('length', 10000), 'NOT NULL'), 97e8827d73SSzymon Olewniczak 'thread_comment_id' => array(array('numeric'), 'NULL'), 988a638198SSzymon Olewniczak 'task_program_id' => array(array('numeric'), 'NULL') 998a638198SSzymon Olewniczak )); 1008a638198SSzymon Olewniczak 1018a638198SSzymon Olewniczak //we've created empty object 1028a638198SSzymon Olewniczak if ($this->id === NULL) { 1038a638198SSzymon Olewniczak $this->original_poster = $this->model->user_nick; 1048a638198SSzymon Olewniczak $this->create_date = date('c'); 1058a638198SSzymon Olewniczak $this->last_activity_date = $this->create_date; 1068a638198SSzymon Olewniczak $this->last_modification_date = $this->create_date; 1078a638198SSzymon Olewniczak 1088a638198SSzymon Olewniczak $this->state = 'opened'; 1098a638198SSzymon Olewniczak 1108a638198SSzymon Olewniczak if (isset($defaults['thread'])) { 1118a638198SSzymon Olewniczak $this->thread = $defaults['thread']; 1128a638198SSzymon Olewniczak $this->thread_id = $this->thread->id; 11353df74e7SSzymon Olewniczak $this->coordinator = $this->thread->coordinator; 1146f380773SSzymon Olewniczak 1156f380773SSzymon Olewniczak if ($this->thread->private == '1') { 1166f380773SSzymon Olewniczak $this->private = '1'; 1176f380773SSzymon Olewniczak } 1186f380773SSzymon Olewniczak 1198a638198SSzymon Olewniczak $this->type = 'correction'; 1208a638198SSzymon Olewniczak 1218a638198SSzymon Olewniczak if (isset($defaults['thread_comment'])) { 1228a638198SSzymon Olewniczak $this->thread_comment = $defaults['thread_comment']; 1238a638198SSzymon Olewniczak $this->thread_comment_id = $this->thread_comment->id; 124e8827d73SSzymon Olewniczak 125e8827d73SSzymon Olewniczak if ($this->thread_comment->type == 'cause_real') { 1268a638198SSzymon Olewniczak $this->type = 'corrective'; 127e8827d73SSzymon Olewniczak } else { 128e8827d73SSzymon Olewniczak $this->type = 'preventive'; 129e8827d73SSzymon Olewniczak } 1308a638198SSzymon Olewniczak } 131ff14b107SSzymon Olewniczak } else { 132ff14b107SSzymon Olewniczak $this->type = 'program'; 1338a638198SSzymon Olewniczak } 134a0cd8c78SSzymon Olewniczak 1356f380773SSzymon Olewniczak 1366f380773SSzymon Olewniczak if ($this->thread_id == '') { 1376f380773SSzymon Olewniczak $this->validator->set_rules(array( 1386f380773SSzymon Olewniczak 'task_program_id' => array(array('numeric'), 'NOT NULL'), 1396f380773SSzymon Olewniczak )); 1406f380773SSzymon Olewniczak //this field is unused in program tasks 1416f380773SSzymon Olewniczak $this->validator->delete_rule('thread_comment_id'); 1426f380773SSzymon Olewniczak } 1436f380773SSzymon Olewniczak 144a0cd8c78SSzymon Olewniczak //everyone can report their own program tasks 145a0cd8c78SSzymon Olewniczak if ($this->type == 'program') { 146a0cd8c78SSzymon Olewniczak $this->acl->grant('content', BEZ_PERMISSION_CHANGE); 147a0cd8c78SSzymon Olewniczak $this->acl->grant('plan_date', BEZ_PERMISSION_CHANGE); 148a0cd8c78SSzymon Olewniczak $this->acl->grant('start_time', BEZ_PERMISSION_CHANGE); 149a0cd8c78SSzymon Olewniczak $this->acl->grant('finish_time', BEZ_PERMISSION_CHANGE); 150a0cd8c78SSzymon Olewniczak $this->acl->grant('all_day_event', BEZ_PERMISSION_CHANGE); 151a0cd8c78SSzymon Olewniczak $this->acl->grant('task_program_id', BEZ_PERMISSION_CHANGE); 152a0cd8c78SSzymon Olewniczak $this->acl->grant('cost', BEZ_PERMISSION_CHANGE); 153a0cd8c78SSzymon Olewniczak } 154a0cd8c78SSzymon Olewniczak 155a0cd8c78SSzymon Olewniczak if ($this->type == 'program' && $this->model->get_level() >= BEZ_AUTH_LEADER) { 156a0cd8c78SSzymon Olewniczak $this->acl->grant('assignee', BEZ_PERMISSION_CHANGE); 157a0cd8c78SSzymon Olewniczak $this->acl->grant('participants', BEZ_PERMISSION_CHANGE); 158a0cd8c78SSzymon Olewniczak } 159a0cd8c78SSzymon Olewniczak 160a0cd8c78SSzymon Olewniczak if ($this->type != 'program' && $this->coordinator == $this->model->user_nick) { 161a0cd8c78SSzymon Olewniczak $this->acl->grant('content', BEZ_PERMISSION_CHANGE); 162a0cd8c78SSzymon Olewniczak $this->acl->grant('plan_date', BEZ_PERMISSION_CHANGE); 163a0cd8c78SSzymon Olewniczak $this->acl->grant('start_time', BEZ_PERMISSION_CHANGE); 164a0cd8c78SSzymon Olewniczak $this->acl->grant('finish_time', BEZ_PERMISSION_CHANGE); 165a0cd8c78SSzymon Olewniczak $this->acl->grant('all_day_event', BEZ_PERMISSION_CHANGE); 166a0cd8c78SSzymon Olewniczak $this->acl->grant('task_program_id', BEZ_PERMISSION_CHANGE); 167a0cd8c78SSzymon Olewniczak $this->acl->grant('cost', BEZ_PERMISSION_CHANGE); 168a0cd8c78SSzymon Olewniczak 169a0cd8c78SSzymon Olewniczak $this->acl->grant('assignee', BEZ_PERMISSION_CHANGE); 170a0cd8c78SSzymon Olewniczak $this->acl->grant('participants', BEZ_PERMISSION_CHANGE); 171a0cd8c78SSzymon Olewniczak } 172a0cd8c78SSzymon Olewniczak 173e8827d73SSzymon Olewniczak //we get object form db 174e8827d73SSzymon Olewniczak } else { 175e8827d73SSzymon Olewniczak if (isset($defaults['thread']) && $this->thread_id == $defaults['thread']->id) { 176e8827d73SSzymon Olewniczak $this->thread = $defaults['thread']; 177e8827d73SSzymon Olewniczak } 178e8827d73SSzymon Olewniczak 179e8827d73SSzymon Olewniczak if (isset($defaults['thread_comment']) && $this->thread_comment_id == $defaults['thread_comment']->id) { 180e8827d73SSzymon Olewniczak $this->thread_comment = $defaults['thread_comment']; 181e8827d73SSzymon Olewniczak } 182e8827d73SSzymon Olewniczak 1836f380773SSzymon Olewniczak if ($this->thread_id == '') { 1846f380773SSzymon Olewniczak $this->validator->set_rules(array( 1856f380773SSzymon Olewniczak 'task_program_id' => array(array('numeric'), 'NOT NULL'), 1866f380773SSzymon Olewniczak )); 1876f380773SSzymon Olewniczak //this field is unused in program tasks 1886f380773SSzymon Olewniczak $this->validator->delete_rule('thread_comment_id'); 1896f380773SSzymon Olewniczak } 1906f380773SSzymon Olewniczak 1916f380773SSzymon Olewniczak //private tasks 1926f380773SSzymon Olewniczak if ($this->model->level < BEZ_AUTH_ADMIN && $this->private == '1') { 1936f380773SSzymon Olewniczak if ($this->get_participant($this->model->user_nick) === false && 1946f380773SSzymon Olewniczak ($this->thread_id != '' && $this->__get('thread')->get_participant($this->model->user_nick) === false)) { 1956f380773SSzymon Olewniczak $this->acl->revoke(self::get_select_columns(), BEZ_AUTH_LEADER); 1966f380773SSzymon Olewniczak return; 1976f380773SSzymon Olewniczak } 1986f380773SSzymon Olewniczak } 1996f380773SSzymon Olewniczak 200a0cd8c78SSzymon Olewniczak //user can close their tasks 201a0cd8c78SSzymon Olewniczak if ($this->assignee == $this->model->user_nick || $this->model->get_level() >= BEZ_AUTH_LEADER) { 202a0cd8c78SSzymon Olewniczak $this->acl->grant('state', BEZ_PERMISSION_CHANGE); 203a0cd8c78SSzymon Olewniczak } 204a0cd8c78SSzymon Olewniczak 205a0cd8c78SSzymon Olewniczak if ($this->type == 'program' && $this->original_poster == $this->model->user_nick) { 206a0cd8c78SSzymon Olewniczak $this->acl->grant('content', BEZ_PERMISSION_CHANGE); 207a0cd8c78SSzymon Olewniczak $this->acl->grant('plan_date', BEZ_PERMISSION_CHANGE); 208a0cd8c78SSzymon Olewniczak $this->acl->grant('start_time', BEZ_PERMISSION_CHANGE); 209a0cd8c78SSzymon Olewniczak $this->acl->grant('finish_time', BEZ_PERMISSION_CHANGE); 210a0cd8c78SSzymon Olewniczak $this->acl->grant('all_day_event', BEZ_PERMISSION_CHANGE); 211a0cd8c78SSzymon Olewniczak $this->acl->grant('task_program_id', BEZ_PERMISSION_CHANGE); 212a0cd8c78SSzymon Olewniczak $this->acl->grant('cost', BEZ_PERMISSION_CHANGE); 213a0cd8c78SSzymon Olewniczak } 214a0cd8c78SSzymon Olewniczak 215a0cd8c78SSzymon Olewniczak if (($this->type != 'program' && $this->coordinator == $this->model->user_nick) || 216a0cd8c78SSzymon Olewniczak ($this->model->get_level() >= BEZ_AUTH_LEADER)) { 217a0cd8c78SSzymon Olewniczak $this->acl->grant('content', BEZ_PERMISSION_CHANGE); 218a0cd8c78SSzymon Olewniczak $this->acl->grant('plan_date', BEZ_PERMISSION_CHANGE); 219a0cd8c78SSzymon Olewniczak $this->acl->grant('start_time', BEZ_PERMISSION_CHANGE); 220a0cd8c78SSzymon Olewniczak $this->acl->grant('finish_time', BEZ_PERMISSION_CHANGE); 221a0cd8c78SSzymon Olewniczak $this->acl->grant('all_day_event', BEZ_PERMISSION_CHANGE); 222a0cd8c78SSzymon Olewniczak $this->acl->grant('task_program_id', BEZ_PERMISSION_CHANGE); 223a0cd8c78SSzymon Olewniczak $this->acl->grant('cost', BEZ_PERMISSION_CHANGE); 224a0cd8c78SSzymon Olewniczak 225a0cd8c78SSzymon Olewniczak $this->acl->grant('assignee', BEZ_PERMISSION_CHANGE); 226a0cd8c78SSzymon Olewniczak $this->acl->grant('participants', BEZ_PERMISSION_CHANGE); 227f7519ef1SSzymon Olewniczak $this->acl->grant('state', BEZ_PERMISSION_CHANGE); 228a0cd8c78SSzymon Olewniczak } 229e8827d73SSzymon Olewniczak } 2308a638198SSzymon Olewniczak } 2318a638198SSzymon Olewniczak 23214a1f0a4SSzymon Olewniczak public function update_virutal() { 23314a1f0a4SSzymon Olewniczak if ($this->state == 'done') { 23414a1f0a4SSzymon Olewniczak $this->priority = ''; 23514a1f0a4SSzymon Olewniczak } else { 23614a1f0a4SSzymon Olewniczak $now = date('Y-m-d'); 23714a1f0a4SSzymon Olewniczak $plus_1_month = date('Y-m-d', strtotime('+1 month')); 23814a1f0a4SSzymon Olewniczak 23914a1f0a4SSzymon Olewniczak if ($this->plan_date >= $plus_1_month) { 24014a1f0a4SSzymon Olewniczak $this->priority = '0'; 24114a1f0a4SSzymon Olewniczak } elseif ($this->plan_date >= $now) { 24214a1f0a4SSzymon Olewniczak $this->priority = '1'; 24314a1f0a4SSzymon Olewniczak } else { 24414a1f0a4SSzymon Olewniczak $this->priority = '2'; 24514a1f0a4SSzymon Olewniczak } 24614a1f0a4SSzymon Olewniczak } 24714a1f0a4SSzymon Olewniczak } 2488a638198SSzymon Olewniczak 2498a638198SSzymon Olewniczak public function set_data($post, $filter=NULL) { 250ff14b107SSzymon Olewniczak //all day event 251ff14b107SSzymon Olewniczak if (!isset($post['all_day_event'])) { 252ff14b107SSzymon Olewniczak $post['all_day_event'] = '0'; 253ff14b107SSzymon Olewniczak } 254ff14b107SSzymon Olewniczak 25553df74e7SSzymon Olewniczak parent::set_data($post); 25653df74e7SSzymon Olewniczak 25753df74e7SSzymon Olewniczak $this->content_html = p_render('xhtml',p_get_instructions($this->content), $ignore); 25853df74e7SSzymon Olewniczak 25953df74e7SSzymon Olewniczak if (!isset($post['assignee'])) { 26053df74e7SSzymon Olewniczak $this->assignee = $this->model->user_nick; 26153df74e7SSzymon Olewniczak } 26253df74e7SSzymon Olewniczak 26353df74e7SSzymon Olewniczak //update dates 26453df74e7SSzymon Olewniczak $this->last_modification_date = date('c'); 26553df74e7SSzymon Olewniczak $this->last_activity_date = $this->last_modification_date; 26653df74e7SSzymon Olewniczak 26714a1f0a4SSzymon Olewniczak //update virtual 26814a1f0a4SSzymon Olewniczak $this->update_virutal(); 26914a1f0a4SSzymon Olewniczak 2708a638198SSzymon Olewniczak return true; 2718a638198SSzymon Olewniczak } 2728a638198SSzymon Olewniczak 273e8827d73SSzymon Olewniczak public function set_state($state) { 274e8827d73SSzymon Olewniczak if ($this->acl_of('state') < BEZ_PERMISSION_CHANGE) { 275e8827d73SSzymon Olewniczak throw new PermissionDeniedException(); 276e8827d73SSzymon Olewniczak } 277e8827d73SSzymon Olewniczak 278e8827d73SSzymon Olewniczak if (!in_array($state, array('opened', 'done'))) { 279e8827d73SSzymon Olewniczak throw new ValidationException('task', array('sholud be opened or done')); 280e8827d73SSzymon Olewniczak } 281e8827d73SSzymon Olewniczak 282e8827d73SSzymon Olewniczak //nothing to do 283e8827d73SSzymon Olewniczak if ($state == $this->state) { 284e8827d73SSzymon Olewniczak return; 285e8827d73SSzymon Olewniczak } 286e8827d73SSzymon Olewniczak 287e8827d73SSzymon Olewniczak if ($state == 'done') { 288e8827d73SSzymon Olewniczak $this->model->sqlite->query("UPDATE {$this->get_table_name()} SET state=?, closed_by=?, close_date=? WHERE id=?", 289e8827d73SSzymon Olewniczak $state, 290e8827d73SSzymon Olewniczak $this->model->user_nick, 291e8827d73SSzymon Olewniczak date('c'), 292e8827d73SSzymon Olewniczak $this->id); 293e8827d73SSzymon Olewniczak //reopen the task 294e8827d73SSzymon Olewniczak } else { 295e8827d73SSzymon Olewniczak $this->model->sqlite->query("UPDATE {$this->get_table_name()} SET state=? WHERE id=?", $state, $this->id); 296e8827d73SSzymon Olewniczak } 297e8827d73SSzymon Olewniczak 298e8827d73SSzymon Olewniczak $this->state = $state; 299e8827d73SSzymon Olewniczak } 300e8827d73SSzymon Olewniczak 301e8827d73SSzymon Olewniczak public function update_last_activity() { 302e8827d73SSzymon Olewniczak $this->last_activity_date = date('c'); 303e8827d73SSzymon Olewniczak $this->model->sqlite->query('UPDATE task SET last_activity_date=? WHERE id=?', 304e8827d73SSzymon Olewniczak $this->last_activity_date, $this->id); 305e8827d73SSzymon Olewniczak } 306e8827d73SSzymon Olewniczak 307f7519ef1SSzymon Olewniczak public function can_add_comments() { 308f7519ef1SSzymon Olewniczak if ($this->thread_id != '' && $this->thread->state == 'closed') { 309f7519ef1SSzymon Olewniczak return false; 310f7519ef1SSzymon Olewniczak } 311f7519ef1SSzymon Olewniczak 312f7519ef1SSzymon Olewniczak if ($this->state == 'opened' || 313f7519ef1SSzymon Olewniczak ($this->state == 'done' && 314f7519ef1SSzymon Olewniczak $this->acl_of('state') >= BEZ_PERMISSION_CHANGE)) { 315f7519ef1SSzymon Olewniczak return true; 316f7519ef1SSzymon Olewniczak } 317f7519ef1SSzymon Olewniczak 318f7519ef1SSzymon Olewniczak return false; 319f7519ef1SSzymon Olewniczak } 320f7519ef1SSzymon Olewniczak 321a0cd8c78SSzymon Olewniczak public function can_add_participants() { 322a0cd8c78SSzymon Olewniczak return in_array($this->state, array('opened')); 323a0cd8c78SSzymon Olewniczak } 324a0cd8c78SSzymon Olewniczak 325e8827d73SSzymon Olewniczak public function get_participants($filter='') { 326e8827d73SSzymon Olewniczak if ($this->id === NULL) { 327e8827d73SSzymon Olewniczak return array(); 328e8827d73SSzymon Olewniczak } 329e8827d73SSzymon Olewniczak 330e8827d73SSzymon Olewniczak $sql = 'SELECT * FROM task_participant WHERE'; 331e8827d73SSzymon Olewniczak $possible_flags = array('original_poster', 'assignee', 'commentator', 'subscribent'); 332e8827d73SSzymon Olewniczak if ($filter != '') { 333e8827d73SSzymon Olewniczak if (!in_array($filter, $possible_flags)) { 334e8827d73SSzymon Olewniczak throw new \Exception("unknown flag $filter"); 335e8827d73SSzymon Olewniczak } 336e8827d73SSzymon Olewniczak $sql .= " $filter=1 AND"; 337e8827d73SSzymon Olewniczak } 338b331b892SSzymon Olewniczak $sql .= ' task_id=? AND removed=0 ORDER BY user_id'; 339e8827d73SSzymon Olewniczak 340e8827d73SSzymon Olewniczak $r = $this->model->sqlite->query($sql, $this->id); 341e8827d73SSzymon Olewniczak $pars = $this->model->sqlite->res2arr($r); 342e8827d73SSzymon Olewniczak $participants = array(); 343e8827d73SSzymon Olewniczak foreach ($pars as $par) { 344e8827d73SSzymon Olewniczak $participants[$par['user_id']] = $par; 345e8827d73SSzymon Olewniczak } 346e8827d73SSzymon Olewniczak 347e8827d73SSzymon Olewniczak return $participants; 348e8827d73SSzymon Olewniczak } 349e8827d73SSzymon Olewniczak 350b331b892SSzymon Olewniczak public function get_participant($user_id, $can_be_removed=false) { 351e8827d73SSzymon Olewniczak if ($this->id === NULL) { 352e8827d73SSzymon Olewniczak return array(); 353e8827d73SSzymon Olewniczak } 354e8827d73SSzymon Olewniczak 355b331b892SSzymon Olewniczak $q = 'SELECT * FROM task_participant WHERE task_id=? AND user_id=?'; 356b331b892SSzymon Olewniczak if (!$can_be_removed) { 357b331b892SSzymon Olewniczak $q .= ' AND removed=0'; 358b331b892SSzymon Olewniczak } 359b331b892SSzymon Olewniczak $r = $this->model->sqlite->query($q, $this->id, $user_id); 360e8827d73SSzymon Olewniczak $par = $this->model->sqlite->res2row($r); 361e8827d73SSzymon Olewniczak if (!is_array($par)) { 362e8827d73SSzymon Olewniczak return false; 363e8827d73SSzymon Olewniczak } 364e8827d73SSzymon Olewniczak 365e8827d73SSzymon Olewniczak return $par; 366e8827d73SSzymon Olewniczak } 367e8827d73SSzymon Olewniczak 368e8827d73SSzymon Olewniczak public function is_subscribent($user_id=null) { 369e8827d73SSzymon Olewniczak if ($user_id == null) { 370e8827d73SSzymon Olewniczak $user_id = $this->model->user_nick; 371e8827d73SSzymon Olewniczak } 372e8827d73SSzymon Olewniczak $par = $this->get_participant($user_id); 373e8827d73SSzymon Olewniczak if ($par['subscribent'] == 1) { 374e8827d73SSzymon Olewniczak return true; 375e8827d73SSzymon Olewniczak } 376e8827d73SSzymon Olewniczak return false; 377e8827d73SSzymon Olewniczak } 378e8827d73SSzymon Olewniczak 379e8827d73SSzymon Olewniczak public function remove_participant_flags($user_id, $flags) { 380e8827d73SSzymon Olewniczak //thread not saved yet 381e8827d73SSzymon Olewniczak if ($this->id === NULL) { 382e8827d73SSzymon Olewniczak throw new \Exception('cannot remove flags from not saved thread'); 383e8827d73SSzymon Olewniczak } 384e8827d73SSzymon Olewniczak 385b331b892SSzymon Olewniczak $participant = $this->get_participant($user_id, true); 386b331b892SSzymon Olewniczak if ($participant === false) { 387b331b892SSzymon Olewniczak throw new ConsistencyViolationException("$user_id isn't participant"); 388b331b892SSzymon Olewniczak } 389b331b892SSzymon Olewniczak 390e8827d73SSzymon Olewniczak $possible_flags = array('original_poster', 'assignee', 'commentator', 'subscribent'); 391e8827d73SSzymon Olewniczak if (array_intersect($flags, $possible_flags) != $flags) { 392e8827d73SSzymon Olewniczak throw new \Exception('unknown flags'); 393e8827d73SSzymon Olewniczak } 394e8827d73SSzymon Olewniczak 395e8827d73SSzymon Olewniczak $set = implode(',', array_map(function ($v) { return "$v=0"; }, $flags)); 396e8827d73SSzymon Olewniczak 397e8827d73SSzymon Olewniczak $sql = "UPDATE task_participant SET $set WHERE task_id=? AND user_id=?"; 398e8827d73SSzymon Olewniczak $this->model->sqlite->query($sql, $this->id, $user_id); 399e8827d73SSzymon Olewniczak 400e8827d73SSzymon Olewniczak } 401e8827d73SSzymon Olewniczak 402e8827d73SSzymon Olewniczak public function set_participant_flags($user_id, $flags=array()) { 403e8827d73SSzymon Olewniczak //thread not saved yet 404e8827d73SSzymon Olewniczak if ($this->id === NULL) { 405e8827d73SSzymon Olewniczak throw new \Exception('cannot add flags to not saved thread'); 406e8827d73SSzymon Olewniczak } 407e8827d73SSzymon Olewniczak 408e8827d73SSzymon Olewniczak //validate user 409e8827d73SSzymon Olewniczak if (!$this->model->userFactory->exists($user_id)) { 410e8827d73SSzymon Olewniczak throw new \Exception("$user_id isn't dokuwiki user"); 411e8827d73SSzymon Olewniczak } 412e8827d73SSzymon Olewniczak 413e8827d73SSzymon Olewniczak $possible_flags = array('original_poster', 'assignee', 'commentator', 'subscribent'); 414e8827d73SSzymon Olewniczak if (array_intersect($flags, $possible_flags) != $flags) { 415e8827d73SSzymon Olewniczak throw new \Exception('unknown flags'); 416e8827d73SSzymon Olewniczak } 417e8827d73SSzymon Olewniczak 418b331b892SSzymon Olewniczak $participant = $this->get_participant($user_id, true); 419e8827d73SSzymon Olewniczak if ($participant == false) { 420e8827d73SSzymon Olewniczak $participant = array_fill_keys($possible_flags, 0); 421e8827d73SSzymon Olewniczak 422e8827d73SSzymon Olewniczak $participant['task_id'] = $this->id; 423e8827d73SSzymon Olewniczak $participant['user_id'] = $user_id; 424e8827d73SSzymon Olewniczak $participant['added_by'] = $this->model->user_nick; 425e8827d73SSzymon Olewniczak $participant['added_date'] = date('c'); 426b331b892SSzymon Olewniczak 427e8827d73SSzymon Olewniczak $values = array_merge($participant, array_fill_keys($flags, 1)); 428b331b892SSzymon Olewniczak $this->model->sqlite->storeEntry('task_participant', $values); 429b331b892SSzymon Olewniczak } else { 430b331b892SSzymon Olewniczak $set = implode(',', array_map(function($flag) { return "$flag=1"; }, $flags)); 431e8827d73SSzymon Olewniczak 432b331b892SSzymon Olewniczak if ($participant['removed'] == '1') { 433b331b892SSzymon Olewniczak $set .= ',removed=0'; 434b331b892SSzymon Olewniczak } 435e8827d73SSzymon Olewniczak 436b331b892SSzymon Olewniczak $q = "UPDATE task_participant SET $set WHERE task_id=? AND user_id=?"; 437b331b892SSzymon Olewniczak $this->model->sqlite->query($q, $this->id, $user_id); 438b331b892SSzymon Olewniczak } 439b331b892SSzymon Olewniczak } 440b331b892SSzymon Olewniczak 441b331b892SSzymon Olewniczak public function remove_participant($user_id) { 442b331b892SSzymon Olewniczak //thread not saved yet 443b331b892SSzymon Olewniczak if ($this->id === NULL) { 444b331b892SSzymon Olewniczak throw new \Exception('cannot remove flags from not saved thread'); 445b331b892SSzymon Olewniczak } 446b331b892SSzymon Olewniczak 447b331b892SSzymon Olewniczak $participant = $this->get_participant($user_id); 448b331b892SSzymon Olewniczak if ($participant === false) { 449b331b892SSzymon Olewniczak throw new ConsistencyViolationException("$user_id isn't participant"); 450b331b892SSzymon Olewniczak } 451b331b892SSzymon Olewniczak 452b331b892SSzymon Olewniczak if ($participant['assignee'] == '1') { 453b331b892SSzymon Olewniczak throw new ConsistencyViolationException("cannot remove assignee"); 454b331b892SSzymon Olewniczak } 455b331b892SSzymon Olewniczak 456b331b892SSzymon Olewniczak $q = "UPDATE task_participant SET removed=1 WHERE task_id=? AND user_id=?"; 457b331b892SSzymon Olewniczak $this->model->sqlite->query($q, $this->id, $user_id); 458b331b892SSzymon Olewniczak 459e8827d73SSzymon Olewniczak } 460e8827d73SSzymon Olewniczak 461*a5de966aSSzymon Olewniczak public function pin($thread_id) { 462*a5de966aSSzymon Olewniczak if ($this->acl_of('thread_id') < BEZ_PERMISSION_CHANGE) { 463*a5de966aSSzymon Olewniczak throw new PermissionDeniedException(); 464*a5de966aSSzymon Olewniczak } 465*a5de966aSSzymon Olewniczak if ($this->thread_id != '') { 466*a5de966aSSzymon Olewniczak throw new ConsistencyViolationException('task already pinned to thread'); 467*a5de966aSSzymon Olewniczak } 468*a5de966aSSzymon Olewniczak 469*a5de966aSSzymon Olewniczak //check if thread exists and isn't closed 470*a5de966aSSzymon Olewniczak $q = "SELECT id FROM thread_view WHERE id = ? AND state IN ('opened', 'done')"; 471*a5de966aSSzymon Olewniczak $r = $this->model->sqlite->query($q, $thread_id); 472*a5de966aSSzymon Olewniczak if ($this->model->sqlite->res2count($r) == 0) { 473*a5de966aSSzymon Olewniczak throw new ValidationException("task", array('thread_id' => 'pin_task')); 474*a5de966aSSzymon Olewniczak } 475*a5de966aSSzymon Olewniczak $q = "UPDATE task SET type='correction', thread_id=? WHERE id=?"; 476*a5de966aSSzymon Olewniczak $this->model->sqlite->query($q, $thread_id, $this->id); 477*a5de966aSSzymon Olewniczak } 478*a5de966aSSzymon Olewniczak 479*a5de966aSSzymon Olewniczak public function unpin() { 480*a5de966aSSzymon Olewniczak if ($this->acl_of('thread_id') < BEZ_PERMISSION_CHANGE) { 481*a5de966aSSzymon Olewniczak throw new PermissionDeniedException(); 482*a5de966aSSzymon Olewniczak } 483*a5de966aSSzymon Olewniczak 484*a5de966aSSzymon Olewniczak $q = "UPDATE task SET type='program', thread_id='' WHERE id=?"; 485*a5de966aSSzymon Olewniczak $this->model->sqlite->query($q, $this->id); 486*a5de966aSSzymon Olewniczak } 487*a5de966aSSzymon Olewniczak 488e8827d73SSzymon Olewniczak public function invite($client) { 489e8827d73SSzymon Olewniczak $this->set_participant_flags($client, array('subscribent')); 490e8827d73SSzymon Olewniczak $this->mail_notify_invite($client); 491e8827d73SSzymon Olewniczak } 492e8827d73SSzymon Olewniczak 49314a1f0a4SSzymon Olewniczak public function mail_notify($content, $users=false, $attachedImages=array()) { 4946f380773SSzymon Olewniczak $mailer = new \Mailer(); 49514a1f0a4SSzymon Olewniczak $mailer->setBody($content, array(), array(), $content, false); 4968a638198SSzymon Olewniczak 4978a638198SSzymon Olewniczak if ($users === FALSE) { 498e8827d73SSzymon Olewniczak $users = $this->get_participants('subscribent'); 4998a638198SSzymon Olewniczak 5008a638198SSzymon Olewniczak //don't notify current user 5018a638198SSzymon Olewniczak unset($users[$this->model->user_nick]); 5028a638198SSzymon Olewniczak } 5038a638198SSzymon Olewniczak 5048a638198SSzymon Olewniczak $emails = array_map(function($user) { 50520e189b9SSzymon Olewniczak if (is_array($user)) { 50620e189b9SSzymon Olewniczak $user = $user['user_id']; 50720e189b9SSzymon Olewniczak } 50820e189b9SSzymon Olewniczak return $this->model->userFactory->get_user_email($user); 5098a638198SSzymon Olewniczak }, $users); 5108a638198SSzymon Olewniczak 5118a638198SSzymon Olewniczak $mailer->to($emails); 51214a1f0a4SSzymon Olewniczak $mailer->subject('#z'.$this->id. ' ' . $this->task_program_name); 5138a638198SSzymon Olewniczak 514a0cd8c78SSzymon Olewniczak //add images 515a0cd8c78SSzymon Olewniczak foreach ($attachedImages as $img) { 516a0cd8c78SSzymon Olewniczak $mailer->attachFile($img['path'], $img['mime'], $img['name'], $img['embed']); 517a0cd8c78SSzymon Olewniczak } 518a0cd8c78SSzymon Olewniczak 5198a638198SSzymon Olewniczak $send = $mailer->send(); 5208a638198SSzymon Olewniczak if ($send === false) { 5218a638198SSzymon Olewniczak //this may mean empty $emails 5228a638198SSzymon Olewniczak //throw new Exception("can't send email"); 5238a638198SSzymon Olewniczak } 5248a638198SSzymon Olewniczak } 5258a638198SSzymon Olewniczak 52614a1f0a4SSzymon Olewniczak public function mail_task_box(&$attachedImages) { 52714a1f0a4SSzymon Olewniczak $tpl = $this->model->action->get_tpl(); 52814a1f0a4SSzymon Olewniczak 52914a1f0a4SSzymon Olewniczak //render style 53014a1f0a4SSzymon Olewniczak $less = new \lessc(); 53114a1f0a4SSzymon Olewniczak $less->addImportDir(DOKU_PLUGIN . 'bez/style/'); 53214a1f0a4SSzymon Olewniczak $style = $less->compileFile(DOKU_PLUGIN . 'bez/style/task.less'); 53314a1f0a4SSzymon Olewniczak 53414a1f0a4SSzymon Olewniczak //render content for mail 53514a1f0a4SSzymon Olewniczak $old_content_html = $this->content_html; 53614a1f0a4SSzymon Olewniczak $this->content_html = p_render('bez_xhtmlmail', p_get_instructions($this->content), $info); 53714a1f0a4SSzymon Olewniczak $attachedImages = array_merge($attachedImages, $info['img']); 53814a1f0a4SSzymon Olewniczak 53914a1f0a4SSzymon Olewniczak $tpl->set('task', $this); 54014a1f0a4SSzymon Olewniczak $tpl->set('style', $style); 54114a1f0a4SSzymon Olewniczak $tpl->set('no_actions', true); 54214a1f0a4SSzymon Olewniczak $task_box = $this->model->action->bez_tpl_include('task_box', true); 54314a1f0a4SSzymon Olewniczak 54414a1f0a4SSzymon Olewniczak $this->content_html = $old_content_html; 54514a1f0a4SSzymon Olewniczak 54614a1f0a4SSzymon Olewniczak return $task_box; 547e8827d73SSzymon Olewniczak } 548e8827d73SSzymon Olewniczak 54914a1f0a4SSzymon Olewniczak public function mail_task(&$attachedImages) { 55014a1f0a4SSzymon Olewniczak $tpl = $this->model->action->get_tpl(); 55114a1f0a4SSzymon Olewniczak 55214a1f0a4SSzymon Olewniczak $task_box = $this->mail_task_box($attachedImages); 55314a1f0a4SSzymon Olewniczak $tpl->set('content', $task_box); 55414a1f0a4SSzymon Olewniczak $content = $this->model->action->bez_tpl_include('mail/task', true); 55514a1f0a4SSzymon Olewniczak 55614a1f0a4SSzymon Olewniczak return $content; 557e8827d73SSzymon Olewniczak } 558e8827d73SSzymon Olewniczak 55914a1f0a4SSzymon Olewniczak public function mail_notify_assignee() { 56014a1f0a4SSzymon Olewniczak $tpl = $this->model->action->get_tpl(); 561e8827d73SSzymon Olewniczak 56214a1f0a4SSzymon Olewniczak //we don't want who 56314a1f0a4SSzymon Olewniczak $tpl->set('who', $this->model->user_nick); 56414a1f0a4SSzymon Olewniczak $tpl->set('action', 'mail_task_assignee'); 56514a1f0a4SSzymon Olewniczak $attachedImages = array(); 56614a1f0a4SSzymon Olewniczak $content = $this->mail_task($attachedImages); 56714a1f0a4SSzymon Olewniczak $this->mail_notify($content, array($this->assignee), $attachedImages); 5688a638198SSzymon Olewniczak } 5698a638198SSzymon Olewniczak 5708a638198SSzymon Olewniczak public function mail_notify_remind($users=false) { 57114a1f0a4SSzymon Olewniczak $tpl = $this->model->action->get_tpl(); 5728a638198SSzymon Olewniczak 57314a1f0a4SSzymon Olewniczak //we don't want who 57414a1f0a4SSzymon Olewniczak $tpl->set('who', ''); 57514a1f0a4SSzymon Olewniczak $tpl->set('action', 'mail_task_remind'); 57614a1f0a4SSzymon Olewniczak $attachedImages = array(); 57714a1f0a4SSzymon Olewniczak $content = $this->mail_task($attachedImages); 57814a1f0a4SSzymon Olewniczak $this->mail_notify($content, $users, $attachedImages); 5798a638198SSzymon Olewniczak } 5808a638198SSzymon Olewniczak 5818a638198SSzymon Olewniczak public function mail_notify_invite($client) { 5828a638198SSzymon Olewniczak $users = array($client); 58314a1f0a4SSzymon Olewniczak $tpl = $this->model->action->get_tpl(); 58414a1f0a4SSzymon Olewniczak 58514a1f0a4SSzymon Olewniczak //we don't want who 58614a1f0a4SSzymon Olewniczak $tpl->set('who', $this->model->user_nick); 58714a1f0a4SSzymon Olewniczak $tpl->set('action', 'mail_task_invite'); 58814a1f0a4SSzymon Olewniczak $attachedImages = array(); 58914a1f0a4SSzymon Olewniczak $content = $this->mail_task($attachedImages); 59014a1f0a4SSzymon Olewniczak $this->mail_notify($content, $users, $attachedImages); 5918a638198SSzymon Olewniczak } 59214a1f0a4SSzymon Olewniczak 59314a1f0a4SSzymon Olewniczak public function mail_notify_change_state($action='') { 59414a1f0a4SSzymon Olewniczak $tpl = $this->model->action->get_tpl(); 59514a1f0a4SSzymon Olewniczak 59614a1f0a4SSzymon Olewniczak $tpl->set('who', $this->model->user_nick); 59714a1f0a4SSzymon Olewniczak $tpl->set('action', $action); 59814a1f0a4SSzymon Olewniczak $attachedImages = array(); 59914a1f0a4SSzymon Olewniczak $content = $this->mail_task($attachedImages); 60014a1f0a4SSzymon Olewniczak $this->mail_notify($content, false, $attachedImages); 60114a1f0a4SSzymon Olewniczak } 60214a1f0a4SSzymon Olewniczak 6038a638198SSzymon Olewniczak} 604