1aa11d6a4Sleibler<?php 2aa11d6a4Sleibler/** 3aa11d6a4Sleibler * DokuWiki Plugin todo_list (Syntax Component) 4aa11d6a4Sleibler * 5aa11d6a4Sleibler * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html 6aa11d6a4Sleibler */ 7aa11d6a4Sleibler 8aa11d6a4Sleibler// must be run within Dokuwiki 9aa11d6a4Sleiblerif(!defined('DOKU_INC')) die(); 10aa11d6a4Sleibler 11aa11d6a4Sleibler/** 12aa11d6a4Sleibler * Class syntax_plugin_todo_list 13aa11d6a4Sleibler */ 14aa11d6a4Sleiblerclass syntax_plugin_todo_list extends syntax_plugin_todo_todo { 15aa11d6a4Sleibler 16aa11d6a4Sleibler /** 17aa11d6a4Sleibler * @return string Syntax mode type 18aa11d6a4Sleibler */ 19aa11d6a4Sleibler public function getType() { 20aa11d6a4Sleibler return 'substition'; 21aa11d6a4Sleibler } 22aa11d6a4Sleibler 23aa11d6a4Sleibler /** 24aa11d6a4Sleibler * @return string Paragraph type 25aa11d6a4Sleibler */ 26aa11d6a4Sleibler public function getPType() { 27aa11d6a4Sleibler return 'block'; 28aa11d6a4Sleibler } 29aa11d6a4Sleibler 30aa11d6a4Sleibler /** 31aa11d6a4Sleibler * @return int Sort order - Low numbers go before high numbers 32aa11d6a4Sleibler */ 33aa11d6a4Sleibler public function getSort() { 34aa11d6a4Sleibler return 250; 35aa11d6a4Sleibler } 36aa11d6a4Sleibler 37aa11d6a4Sleibler /** 38aa11d6a4Sleibler * Connect lookup pattern to lexer. 39aa11d6a4Sleibler * 40aa11d6a4Sleibler * @param string $mode Parser mode 41aa11d6a4Sleibler */ 42aa11d6a4Sleibler public function connectTo($mode) { 43aa11d6a4Sleibler $this->Lexer->addSpecialPattern('~~TODOLIST[^~]*~~', $mode, 'plugin_todo_list'); 44aa11d6a4Sleibler } 45aa11d6a4Sleibler 46aa11d6a4Sleibler /** 47aa11d6a4Sleibler * Handle matches of the todolist syntax 48aa11d6a4Sleibler * 49aa11d6a4Sleibler * @param string $match The match of the syntax 50aa11d6a4Sleibler * @param int $state The state of the handler 51aa11d6a4Sleibler * @param int $pos The position in the document 52aa11d6a4Sleibler * @param Doku_Handler $handler The handler 53aa11d6a4Sleibler * @return array Data for the renderer 54aa11d6a4Sleibler */ 55aa11d6a4Sleibler public function handle($match, $state, $pos, Doku_Handler &$handler) { 56aa11d6a4Sleibler 57aa11d6a4Sleibler $options = substr($match, 10, -2); // strip markup 58aa11d6a4Sleibler $options = explode(' ', $options); 597eccb63fSrunout-at $data = array( 6061e401ffSrunout-at 'header' => $this->getConf("Header"), 61aa11d6a4Sleibler 'completed' => 'all', 6296468fefSrunout-at 'assigned' => 'all', 63f1a46b72SMax Westen 'ns' => 'all', 64819065edSrunout-at 'showdate' => $this->getConf("ShowdateList"), 6561e401ffSrunout-at 'checkbox' => $this->getConf("Checkbox"), 665a968c91Srunout-at 'username' => $this->getConf("Username"), 67aa11d6a4Sleibler ); 68aa11d6a4Sleibler $allowedvalues = array('yes', 'no'); 69aa11d6a4Sleibler foreach($options as $option) { 70aa11d6a4Sleibler @list($key, $value) = explode(':', $option, 2); 710e738436Srunout-at switch($key) { 720e738436Srunout-at case 'header': // how should the header be rendered? 7351046ae0Srunout-at if(in_array($value, array('id', 'firstheader', 'none'))) { 74bce03c8aSrunout-at $data['header'] = $value; 75bce03c8aSrunout-at } 76bce03c8aSrunout-at break; 77780010cdSrunout-at case 'showdate': 78780010cdSrunout-at if(in_array($value, $allowedvalues)) { 79819065edSrunout-at $data['showdate'] = ($value == 'yes'); 80780010cdSrunout-at } 81780010cdSrunout-at break; 8296468fefSrunout-at case 'checkbox': // should checkbox be rendered? 8396468fefSrunout-at if(in_array($value, $allowedvalues)) { 8496468fefSrunout-at $data['checkbox'] = ($value == 'yes'); 8596468fefSrunout-at } 8696468fefSrunout-at break; 87aa11d6a4Sleibler case 'completed': 88aa11d6a4Sleibler if(in_array($value, $allowedvalues)) { 89aa11d6a4Sleibler $data['completed'] = ($value == 'yes'); 90aa11d6a4Sleibler } 91aa11d6a4Sleibler break; 9296468fefSrunout-at case 'username': // how should the username be rendered? 9396468fefSrunout-at if(in_array($value, array('user', 'real', 'none'))) { 9496468fefSrunout-at $data['username'] = $value; 9596468fefSrunout-at } 9696468fefSrunout-at break; 97aa11d6a4Sleibler case 'assigned': 98aa11d6a4Sleibler if(in_array($value, $allowedvalues)) { 99aa11d6a4Sleibler $data['assigned'] = ($value == 'yes'); 100aa11d6a4Sleibler break; 101aa11d6a4Sleibler } 102aa11d6a4Sleibler //assigned? 103aa11d6a4Sleibler $data['assigned'] = explode(',', $value); 104e0ec7364Sleibler // @date 20140317 le: if check for logged in user, also check for logged in user email address 105e0ec7364Sleibler if( in_array( '@@USER@@', $data['assigned'] ) ) { 10664d3f721Sleibler $data['assigned'][] = '@@MAIL@@'; 107e0ec7364Sleibler } 108e1d9ed71Seinhirn $data['assigned'] = array_map( array($this,"__todolistTrimUser"), $data['assigned'] ); 109aa11d6a4Sleibler break; 110f1a46b72SMax Westen case 'ns': 111f1a46b72SMax Westen $data['ns'] = $value; 112f1a46b72SMax Westen break; 1131a73e155Srunout-at case 'startbefore': 114a9e1335fSrunout-at list($data['startbefore'], $data['startignore']) = $this->analyseDate($value); 1151a73e155Srunout-at break; 1161a73e155Srunout-at case 'startafter': 117a9e1335fSrunout-at list($data['startafter'], $data['startignore']) = $this->analyseDate($value); 1181a73e155Srunout-at break; 1191a73e155Srunout-at case 'duebefore': 120a9e1335fSrunout-at list($data['duebefore'], $data['dueignore']) = $this->analyseDate($value); 1211a73e155Srunout-at break; 1221a73e155Srunout-at case 'dueafter': 123a9e1335fSrunout-at list($data['dueafter'], $data['dueignore']) = $this->analyseDate($value); 1241a73e155Srunout-at break; 125aa11d6a4Sleibler } 126aa11d6a4Sleibler } 127aa11d6a4Sleibler return $data; 128aa11d6a4Sleibler } 129aa11d6a4Sleibler 130aa11d6a4Sleibler /** 131aa11d6a4Sleibler * Render xhtml output or metadata 132aa11d6a4Sleibler * 133aa11d6a4Sleibler * @param string $mode Renderer mode (supported modes: xhtml) 134aa11d6a4Sleibler * @param Doku_Renderer $renderer The renderer 135aa11d6a4Sleibler * @param array $data The data from the handler() function 136aa11d6a4Sleibler * @return bool If rendering was successful. 137aa11d6a4Sleibler */ 138aa11d6a4Sleibler public function render($mode, Doku_Renderer &$renderer, $data) { 139aa11d6a4Sleibler global $conf; 140aa11d6a4Sleibler 141aa11d6a4Sleibler if($mode != 'xhtml') return false; 142aa11d6a4Sleibler /** @var Doku_Renderer_xhtml $renderer */ 143aa11d6a4Sleibler 144aa11d6a4Sleibler $opts['pattern'] = '/<todo([^>]*)>(.*)<\/todo[\W]*?>/'; //all todos in a wiki page 145aa11d6a4Sleibler //TODO check if storing subpatterns doesn't cost too much resources 146aa11d6a4Sleibler 147aa11d6a4Sleibler // search(&$data, $base, $func, $opts,$dir='',$lvl=1,$sort='natural') 148aa11d6a4Sleibler search($todopages, $conf['datadir'], array($this, 'search_todos'), $opts); //browse wiki pages with callback to search_pattern 149aa11d6a4Sleibler 150aa11d6a4Sleibler $todopages = $this->filterpages($todopages, $data); 151aa11d6a4Sleibler 15296468fefSrunout-at $this->htmlTodoTable($renderer, $todopages, $data); 153aa11d6a4Sleibler 154aa11d6a4Sleibler return true; 155aa11d6a4Sleibler } 156aa11d6a4Sleibler 157aa11d6a4Sleibler /** 158aa11d6a4Sleibler * Custom search callback 159aa11d6a4Sleibler * 160aa11d6a4Sleibler * This function is called for every found file or 161aa11d6a4Sleibler * directory. When a directory is given to the function it has to 162aa11d6a4Sleibler * decide if this directory should be traversed (true) or not (false). 163aa11d6a4Sleibler * Return values for files are ignored 164aa11d6a4Sleibler * 165aa11d6a4Sleibler * All functions should check the ACL for document READ rights 166aa11d6a4Sleibler * namespaces (directories) are NOT checked (when sneaky_index is 0) as this 167aa11d6a4Sleibler * would break the recursion (You can have an nonreadable dir over a readable 168aa11d6a4Sleibler * one deeper nested) also make sure to check the file type (for example 169aa11d6a4Sleibler * in case of lockfiles). 170aa11d6a4Sleibler * 171aa11d6a4Sleibler * @param array &$data - Reference to the result data structure 172aa11d6a4Sleibler * @param string $base - Base usually $conf['datadir'] 173aa11d6a4Sleibler * @param string $file - current file or directory relative to $base 174aa11d6a4Sleibler * @param string $type - Type either 'd' for directory or 'f' for file 175aa11d6a4Sleibler * @param int $lvl - Current recursion depht 176aa11d6a4Sleibler * @param array $opts - option array as given to search() 177aa11d6a4Sleibler * @return bool if this directory should be traversed (true) or not (false). Return values for files are ignored. 178aa11d6a4Sleibler */ 179aa11d6a4Sleibler public function search_todos(&$data, $base, $file, $type, $lvl, $opts) { 180aa11d6a4Sleibler $item['id'] = pathID($file); //get current file ID 181aa11d6a4Sleibler 182aa11d6a4Sleibler //we do nothing with directories 183aa11d6a4Sleibler if($type == 'd') return true; 184aa11d6a4Sleibler 185aa11d6a4Sleibler //only search txt files 186aa11d6a4Sleibler if(substr($file, -4) != '.txt') return true; 187aa11d6a4Sleibler 188aa11d6a4Sleibler //check ACL 189aa11d6a4Sleibler if(auth_quickaclcheck($item['id']) < AUTH_READ) return false; 190aa11d6a4Sleibler 191aa11d6a4Sleibler $wikitext = rawWiki($item['id']); //get wiki text 192aa11d6a4Sleibler 193aa11d6a4Sleibler $item['count'] = preg_match_all($opts['pattern'], $wikitext, $matches); //count how many times appears the pattern 194aa11d6a4Sleibler if(!empty($item['count'])) { //if it appears at least once 195aa11d6a4Sleibler $item['matches'] = $matches; 196aa11d6a4Sleibler $data[] = $item; 197aa11d6a4Sleibler } 198aa11d6a4Sleibler return true; 199aa11d6a4Sleibler } 200aa11d6a4Sleibler 201aa11d6a4Sleibler /** 2020441172cSeinhirn * Expand assignee-placeholders 2030441172cSeinhirn * 2040441172cSeinhirn * @param $user String to be worked on 2050441172cSeinhirn * @return expanded string 2060441172cSeinhirn */ 2070441172cSeinhirn private function __todolistExpandAssignees($user) { 2080441172cSeinhirn global $USERINFO; 2090441172cSeinhirn if($user == '@@USER@@' && !empty($_SERVER['REMOTE_USER'])) { //$INPUT->server->str('REMOTE_USER') 2100441172cSeinhirn return $_SERVER['REMOTE_USER']; 2110441172cSeinhirn } 2120441172cSeinhirn // @date 20140317 le: check for logged in user email address 2130441172cSeinhirn if( $user == '@@MAIL@@' && isset( $USERINFO['mail'] ) ) { 2140441172cSeinhirn return $USERINFO['mail']; 2150441172cSeinhirn } 2160441172cSeinhirn return $user; 2170441172cSeinhirn } 2180441172cSeinhirn 2190441172cSeinhirn /** 220e1d9ed71Seinhirn * Trim input if it's a user 221e1d9ed71Seinhirn * 222e1d9ed71Seinhirn * @param $user String to be worked on 223e1d9ed71Seinhirn * @return trimmed string 224e1d9ed71Seinhirn */ 225e1d9ed71Seinhirn private function __todolistTrimUser($user) { 226e1d9ed71Seinhirn //placeholder (inspired by replacement-patterns - see https://www.dokuwiki.org/namespace_templates#replacement_patterns) 227e1d9ed71Seinhirn if( $user == '@@USER@@' || $user == '@@MAIL@@' ) { 228e1d9ed71Seinhirn return $user; 229e1d9ed71Seinhirn } 230e1d9ed71Seinhirn //user 231e1d9ed71Seinhirn return trim(ltrim($user, '@')); 232e1d9ed71Seinhirn } 233e1d9ed71Seinhirn 234e1d9ed71Seinhirn /** 235aa11d6a4Sleibler * filter the pages 236aa11d6a4Sleibler * 237aa11d6a4Sleibler * @param $todopages array pages with all todoitems 238aa11d6a4Sleibler * @param $data array listing parameters 239aa11d6a4Sleibler * @return array filtered pages 240aa11d6a4Sleibler */ 241aa11d6a4Sleibler private function filterpages($todopages, $data) { 242aa11d6a4Sleibler $pages = array(); 243aa11d6a4Sleibler foreach($todopages as $page) { 244f1a46b72SMax Westen $parsepage = 0; 245f1a46b72SMax Westen if ($data['ns'] == 'all') { 246f1a46b72SMax Westen // Always return the todo pages 247f1a46b72SMax Westen $parsepage = 1; 248f1a46b72SMax Westen } elseif ($data['ns'] == '/') { 249f1a46b72SMax Westen // Only return the todo page if it's in the root namespace 250f1a46b72SMax Westen if (strpos($page['id'], ':') === FALSE) $parsepage = 1; 251f1a46b72SMax Westen } elseif (substr( $page['id'], 0, strlen($data['ns']) ) === $data['ns']) { 252f1a46b72SMax Westen // Only return the todo page if it starts with the given string 253f1a46b72SMax Westen $parsepage = 1; 254f1a46b72SMax Westen } 255f1a46b72SMax Westen if ($parsepage == 1) { 256aa11d6a4Sleibler $todos = array(); 257aa11d6a4Sleibler // contains 3 arrays: an array with complete matches and 2 arrays with subpatterns 258aa11d6a4Sleibler foreach($page['matches'][1] as $todoindex => $todomatch) { 2599c3e92beSrunout-at $todo = array_merge(array('todotitle' => trim($page['matches'][2][$todoindex]), 'todoindex' => $todoindex), $this->parseTodoArgs($todomatch), $data); 260aa11d6a4Sleibler 2619c3e92beSrunout-at if($this->isRequestedTodo($todo)) { $todos[] = $todo; } 262aa11d6a4Sleibler } 263aa11d6a4Sleibler if(count($todos) > 0) { 264aa11d6a4Sleibler $pages[] = array('id' => $page['id'], 'todos' => $todos); 265aa11d6a4Sleibler } 266aa11d6a4Sleibler } 267f1a46b72SMax Westen } 268aa11d6a4Sleibler return $pages; 269aa11d6a4Sleibler } 270aa11d6a4Sleibler 271aa11d6a4Sleibler /** 272aa11d6a4Sleibler * Create html for table with todos 273aa11d6a4Sleibler * 274aa11d6a4Sleibler * @param Doku_Renderer_xhtml $R 275aa11d6a4Sleibler * @param array $todopages 27696468fefSrunout-at * @param array $data array with rendering options 277aa11d6a4Sleibler */ 27896468fefSrunout-at private function htmlTodoTable($R, $todopages, $data) { 279aa11d6a4Sleibler $R->table_open(); 280aa11d6a4Sleibler foreach($todopages as $page) { 281780e8f87Srunout-at if ($data['header']!='none') { 282aa11d6a4Sleibler $R->tablerow_open(); 283aa11d6a4Sleibler $R->tableheader_open(); 284bce03c8aSrunout-at $R->internallink($page['id'], ($data['header']=='firstheader' ? p_get_first_heading($page['id']) : $page['id'])); 285aa11d6a4Sleibler $R->tableheader_close(); 286aa11d6a4Sleibler $R->tablerow_close(); 287780e8f87Srunout-at } 288aa11d6a4Sleibler foreach($page['todos'] as $todo) { 2899c3e92beSrunout-at//echo "<pre>";var_dump($todo);echo "</pre>"; 290aa11d6a4Sleibler $R->tablerow_open(); 291aa11d6a4Sleibler $R->tablecell_open(); 2925a968c91Srunout-at $R->doc .= $this->createTodoItem($R, $page['id'], array_merge($todo, $data)); 293aa11d6a4Sleibler $R->tablecell_close(); 294aa11d6a4Sleibler $R->tablerow_close(); 295aa11d6a4Sleibler } 296aa11d6a4Sleibler } 297aa11d6a4Sleibler $R->table_close(); 298aa11d6a4Sleibler } 299aa11d6a4Sleibler 300aa11d6a4Sleibler /** 301aa11d6a4Sleibler * Check the conditions for adding a todoitem 302aa11d6a4Sleibler * 303aa11d6a4Sleibler * @param $data array the defined filters 304aa11d6a4Sleibler * @param $checked bool completion status of task; true: finished, false: open 305aa11d6a4Sleibler * @param $todouser string user username of user 306aa11d6a4Sleibler * @return bool if the todoitem should be listed 307aa11d6a4Sleibler */ 30864495bd0Srunout-at /** 30964495bd0Srunout-at * Check the conditions for adding a todoitem 31064495bd0Srunout-at * 31164495bd0Srunout-at * @param $data array the defined filters 31264495bd0Srunout-at * @param $checked bool completion status of task; true: finished, false: open 31364495bd0Srunout-at * @param $todouser string user username of user 31464495bd0Srunout-at * @return bool if the todoitem should be listed 31564495bd0Srunout-at */ 3169c3e92beSrunout-at private function isRequestedTodo($data) { 317aa11d6a4Sleibler //completion status 318aa11d6a4Sleibler $condition1 = $data['completed'] === 'all' //all 3199c3e92beSrunout-at || $data['completed'] === $data['checked']; //yes or no 320aa11d6a4Sleibler 321aa11d6a4Sleibler // resolve placeholder in assignees 322aa11d6a4Sleibler $requestedassignees = array(); 323aa11d6a4Sleibler if(is_array($data['assigned'])) { 3240441172cSeinhirn $requestedassignees = array_map( array($this,"__todolistExpandAssignees"), $data['assigned'] ); 325aa11d6a4Sleibler } 326aa11d6a4Sleibler //assigned 3279c3e92beSrunout-at $condition2 = $condition2 3289c3e92beSrunout-at || $data['assigned'] === 'all' //all 3299c3e92beSrunout-at || (is_bool($data['assigned']) && $data['assigned'] == $data['todouser']); //yes or no 3309c3e92beSrunout-at 3319c3e92beSrunout-at if (!$condition2 && is_array($data['assigned']) && is_array($data['todousers'])) 3329c3e92beSrunout-at foreach($data['todousers'] as $todouser) { 3339c3e92beSrunout-at if(in_array($todouser, $requestedassignees)) { $condition2 = true; break; } 3349c3e92beSrunout-at } 335aa11d6a4Sleibler 3361a73e155Srunout-at //compare start/due dates 337a9e1335fSrunout-at if($condition1 && $condition2) { 338a9e1335fSrunout-at $condition3s = true; $condition3d = true; 33964495bd0Srunout-at if(isset($data['startbefore']) || isset($data['startafter'])) { 34064495bd0Srunout-at if(is_object($data['start'])) { 341*25916678Srunout-at if($data['startignore'] != '!') { 342*25916678Srunout-at if(isset($data['startbefore'])) { $condition3s = $condition3s && new DateTime($data['startbefore']) > $data['start']; } 343*25916678Srunout-at if(isset($data['startafter'])) { $condition3s = $condition3s && new DateTime($data['startafter']) < $data['start']; } 344*25916678Srunout-at } 34564495bd0Srunout-at } else { 34664495bd0Srunout-at if(!$data['startignore'] == '*') { $condition3s = false; } 34764495bd0Srunout-at if($data['startignore'] == '!') { $condition3s = false; } 34864495bd0Srunout-at } 349aa11d6a4Sleibler } 350a9e1335fSrunout-at 35164495bd0Srunout-at if(isset($data['duebefore']) || isset($data['dueafter'])) { 35264495bd0Srunout-at if(is_object($data['due'])) { 353*25916678Srunout-at if($data['dueignore'] != '!') { 354*25916678Srunout-at if(isset($data['duebefore'])) { $condition3d = $condition3d && new DateTime($data['duebefore']) > $data['due']; } 355*25916678Srunout-at if(isset($data['dueafter'])) { $condition3d = $condition3d && new DateTime($data['dueafter']) < $data['due']; } 356*25916678Srunout-at } 35764495bd0Srunout-at } else { 35864495bd0Srunout-at if(!$data['dueignore'] == '*') { $condition3d = false; } 35964495bd0Srunout-at if($data['dueignore'] == '!') { $condition3d = false; } 36064495bd0Srunout-at } 361a9e1335fSrunout-at } 362a9e1335fSrunout-at 363*25916678Srunout-at 364a9e1335fSrunout-at $condition3 = $condition3s && $condition3d; 365a9e1335fSrunout-at } 366a9e1335fSrunout-at 3671a73e155Srunout-at return $condition1 AND $condition2 AND $condition3; 368aa11d6a4Sleibler } 3690d4a1053Srunout-at 37064495bd0Srunout-at 3711a73e155Srunout-at /** 3721a73e155Srunout-at * Analyse of relative/absolute Date and return an absolute date 3731a73e155Srunout-at * 3741a73e155Srunout-at * @param $date string absolute/relative value of the date to analyse 375a9e1335fSrunout-at * @return array absolute date or actual date if $date is invalid 3761a73e155Srunout-at */ 3771a73e155Srunout-at private function analyseDate($date) { 378a9e1335fSrunout-at $result = array($date, ''); 3791a73e155Srunout-at if(is_string($date)) { 38064495bd0Srunout-at if($date == '!') { 38164495bd0Srunout-at $result = array('', '!'); 38264495bd0Srunout-at } elseif ($date =='*') { 38364495bd0Srunout-at $result = array('', '*'); 38464495bd0Srunout-at } else { 38564495bd0Srunout-at if(substr($date, -1) == '*') { 386a9e1335fSrunout-at $date = substr($date, 0, -1); 38764495bd0Srunout-at $result = array($date, '*'); 388a9e1335fSrunout-at } 38964495bd0Srunout-at 3901a73e155Srunout-at if(date('Y-m-d', strtotime($date)) == $date) { 391a9e1335fSrunout-at $result[0] = $date; 3921a73e155Srunout-at } elseif(preg_match('/^[\+\-]\d+$/', $date)) { // check if we have a valid relative value 3931a73e155Srunout-at $newdate = date_create(date('Y-m-d')); 3941a73e155Srunout-at date_modify($newdate, $date . ' day'); 395a9e1335fSrunout-at $result[0] = date_format($newdate, 'Y-m-d'); 3961a73e155Srunout-at } else { 397a9e1335fSrunout-at $result[0] = date('Y-m-d'); 3981a73e155Srunout-at } 3991a73e155Srunout-at } 40064495bd0Srunout-at } else { $result[0] = date('Y-m-d'); } 40164495bd0Srunout-at 4021a73e155Srunout-at return $result; 4031a73e155Srunout-at } 404a9e1335fSrunout-at 40564495bd0Srunout-at 4061a73e155Srunout-at} 407