1*aa11d6a4Sleibler<?php 2*aa11d6a4Sleibler/** 3*aa11d6a4Sleibler * DokuWiki Plugin todo_list (Syntax Component) 4*aa11d6a4Sleibler * 5*aa11d6a4Sleibler * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html 6*aa11d6a4Sleibler */ 7*aa11d6a4Sleibler 8*aa11d6a4Sleibler// must be run within Dokuwiki 9*aa11d6a4Sleiblerif(!defined('DOKU_INC')) die(); 10*aa11d6a4Sleibler 11*aa11d6a4Sleibler/** 12*aa11d6a4Sleibler * Class syntax_plugin_todo_list 13*aa11d6a4Sleibler */ 14*aa11d6a4Sleiblerclass syntax_plugin_todo_list extends syntax_plugin_todo_todo { 15*aa11d6a4Sleibler 16*aa11d6a4Sleibler /** 17*aa11d6a4Sleibler * @return string Syntax mode type 18*aa11d6a4Sleibler */ 19*aa11d6a4Sleibler public function getType() { 20*aa11d6a4Sleibler return 'substition'; 21*aa11d6a4Sleibler } 22*aa11d6a4Sleibler 23*aa11d6a4Sleibler /** 24*aa11d6a4Sleibler * @return string Paragraph type 25*aa11d6a4Sleibler */ 26*aa11d6a4Sleibler public function getPType() { 27*aa11d6a4Sleibler return 'block'; 28*aa11d6a4Sleibler } 29*aa11d6a4Sleibler 30*aa11d6a4Sleibler /** 31*aa11d6a4Sleibler * @return int Sort order - Low numbers go before high numbers 32*aa11d6a4Sleibler */ 33*aa11d6a4Sleibler public function getSort() { 34*aa11d6a4Sleibler return 250; 35*aa11d6a4Sleibler } 36*aa11d6a4Sleibler 37*aa11d6a4Sleibler /** 38*aa11d6a4Sleibler * Connect lookup pattern to lexer. 39*aa11d6a4Sleibler * 40*aa11d6a4Sleibler * @param string $mode Parser mode 41*aa11d6a4Sleibler */ 42*aa11d6a4Sleibler public function connectTo($mode) { 43*aa11d6a4Sleibler $this->Lexer->addSpecialPattern('~~TODOLIST[^~]*~~', $mode, 'plugin_todo_list'); 44*aa11d6a4Sleibler } 45*aa11d6a4Sleibler 46*aa11d6a4Sleibler /** 47*aa11d6a4Sleibler * Handle matches of the todolist syntax 48*aa11d6a4Sleibler * 49*aa11d6a4Sleibler * @param string $match The match of the syntax 50*aa11d6a4Sleibler * @param int $state The state of the handler 51*aa11d6a4Sleibler * @param int $pos The position in the document 52*aa11d6a4Sleibler * @param Doku_Handler $handler The handler 53*aa11d6a4Sleibler * @return array Data for the renderer 54*aa11d6a4Sleibler */ 55*aa11d6a4Sleibler public function handle($match, $state, $pos, Doku_Handler &$handler) { 56*aa11d6a4Sleibler 57*aa11d6a4Sleibler $options = substr($match, 10, -2); // strip markup 58*aa11d6a4Sleibler $options = explode(' ', $options); 59*aa11d6a4Sleibler $data = array( 60*aa11d6a4Sleibler 'completed' => 'all', 61*aa11d6a4Sleibler 'assigned' => 'all' 62*aa11d6a4Sleibler ); 63*aa11d6a4Sleibler $allowedvalues = array('yes', 'no'); 64*aa11d6a4Sleibler foreach($options as $option) { 65*aa11d6a4Sleibler @list($key, $value) = explode(':', $option, 2); 66*aa11d6a4Sleibler switch($key) { 67*aa11d6a4Sleibler case 'completed': 68*aa11d6a4Sleibler if(in_array($value, $allowedvalues)) { 69*aa11d6a4Sleibler $data['completed'] = ($value == 'yes'); 70*aa11d6a4Sleibler } 71*aa11d6a4Sleibler break; 72*aa11d6a4Sleibler case 'assigned': 73*aa11d6a4Sleibler if(in_array($value, $allowedvalues)) { 74*aa11d6a4Sleibler $data['assigned'] = ($value == 'yes'); 75*aa11d6a4Sleibler break; 76*aa11d6a4Sleibler } 77*aa11d6a4Sleibler //assigned? 78*aa11d6a4Sleibler $data['assigned'] = explode(',', $value); 79*aa11d6a4Sleibler $data['assigned'] = array_map( 80*aa11d6a4Sleibler function ($user) { 81*aa11d6a4Sleibler //placeholder 82*aa11d6a4Sleibler if($user == '@@USER@@') { 83*aa11d6a4Sleibler return $user; 84*aa11d6a4Sleibler } 85*aa11d6a4Sleibler //user 86*aa11d6a4Sleibler return ltrim($user, '@'); 87*aa11d6a4Sleibler }, $data['assigned'] 88*aa11d6a4Sleibler ); 89*aa11d6a4Sleibler break; 90*aa11d6a4Sleibler } 91*aa11d6a4Sleibler } 92*aa11d6a4Sleibler return $data; 93*aa11d6a4Sleibler } 94*aa11d6a4Sleibler 95*aa11d6a4Sleibler /** 96*aa11d6a4Sleibler * Render xhtml output or metadata 97*aa11d6a4Sleibler * 98*aa11d6a4Sleibler * @param string $mode Renderer mode (supported modes: xhtml) 99*aa11d6a4Sleibler * @param Doku_Renderer $renderer The renderer 100*aa11d6a4Sleibler * @param array $data The data from the handler() function 101*aa11d6a4Sleibler * @return bool If rendering was successful. 102*aa11d6a4Sleibler */ 103*aa11d6a4Sleibler public function render($mode, Doku_Renderer &$renderer, $data) { 104*aa11d6a4Sleibler global $conf; 105*aa11d6a4Sleibler 106*aa11d6a4Sleibler if($mode != 'xhtml') return false; 107*aa11d6a4Sleibler /** @var Doku_Renderer_xhtml $renderer */ 108*aa11d6a4Sleibler 109*aa11d6a4Sleibler $opts['pattern'] = '/<todo([^>]*)>(.*)<\/todo[\W]*?>/'; //all todos in a wiki page 110*aa11d6a4Sleibler //TODO check if storing subpatterns doesn't cost too much resources 111*aa11d6a4Sleibler 112*aa11d6a4Sleibler // search(&$data, $base, $func, $opts,$dir='',$lvl=1,$sort='natural') 113*aa11d6a4Sleibler search($todopages, $conf['datadir'], array($this, 'search_todos'), $opts); //browse wiki pages with callback to search_pattern 114*aa11d6a4Sleibler 115*aa11d6a4Sleibler $todopages = $this->filterpages($todopages, $data); 116*aa11d6a4Sleibler 117*aa11d6a4Sleibler $this->htmlTodoTable($renderer, $todopages); 118*aa11d6a4Sleibler 119*aa11d6a4Sleibler return true; 120*aa11d6a4Sleibler } 121*aa11d6a4Sleibler 122*aa11d6a4Sleibler /** 123*aa11d6a4Sleibler * Custom search callback 124*aa11d6a4Sleibler * 125*aa11d6a4Sleibler * This function is called for every found file or 126*aa11d6a4Sleibler * directory. When a directory is given to the function it has to 127*aa11d6a4Sleibler * decide if this directory should be traversed (true) or not (false). 128*aa11d6a4Sleibler * Return values for files are ignored 129*aa11d6a4Sleibler * 130*aa11d6a4Sleibler * All functions should check the ACL for document READ rights 131*aa11d6a4Sleibler * namespaces (directories) are NOT checked (when sneaky_index is 0) as this 132*aa11d6a4Sleibler * would break the recursion (You can have an nonreadable dir over a readable 133*aa11d6a4Sleibler * one deeper nested) also make sure to check the file type (for example 134*aa11d6a4Sleibler * in case of lockfiles). 135*aa11d6a4Sleibler * 136*aa11d6a4Sleibler * @param array &$data - Reference to the result data structure 137*aa11d6a4Sleibler * @param string $base - Base usually $conf['datadir'] 138*aa11d6a4Sleibler * @param string $file - current file or directory relative to $base 139*aa11d6a4Sleibler * @param string $type - Type either 'd' for directory or 'f' for file 140*aa11d6a4Sleibler * @param int $lvl - Current recursion depht 141*aa11d6a4Sleibler * @param array $opts - option array as given to search() 142*aa11d6a4Sleibler * @return bool if this directory should be traversed (true) or not (false). Return values for files are ignored. 143*aa11d6a4Sleibler */ 144*aa11d6a4Sleibler public function search_todos(&$data, $base, $file, $type, $lvl, $opts) { 145*aa11d6a4Sleibler $item['id'] = pathID($file); //get current file ID 146*aa11d6a4Sleibler 147*aa11d6a4Sleibler //we do nothing with directories 148*aa11d6a4Sleibler if($type == 'd') return true; 149*aa11d6a4Sleibler 150*aa11d6a4Sleibler //only search txt files 151*aa11d6a4Sleibler if(substr($file, -4) != '.txt') return true; 152*aa11d6a4Sleibler 153*aa11d6a4Sleibler //check ACL 154*aa11d6a4Sleibler if(auth_quickaclcheck($item['id']) < AUTH_READ) return false; 155*aa11d6a4Sleibler 156*aa11d6a4Sleibler $wikitext = rawWiki($item['id']); //get wiki text 157*aa11d6a4Sleibler 158*aa11d6a4Sleibler $item['count'] = preg_match_all($opts['pattern'], $wikitext, $matches); //count how many times appears the pattern 159*aa11d6a4Sleibler if(!empty($item['count'])) { //if it appears at least once 160*aa11d6a4Sleibler $item['matches'] = $matches; 161*aa11d6a4Sleibler $data[] = $item; 162*aa11d6a4Sleibler } 163*aa11d6a4Sleibler return true; 164*aa11d6a4Sleibler } 165*aa11d6a4Sleibler 166*aa11d6a4Sleibler /** 167*aa11d6a4Sleibler * filter the pages 168*aa11d6a4Sleibler * 169*aa11d6a4Sleibler * @param $todopages array pages with all todoitems 170*aa11d6a4Sleibler * @param $data array listing parameters 171*aa11d6a4Sleibler * @return array filtered pages 172*aa11d6a4Sleibler */ 173*aa11d6a4Sleibler private function filterpages($todopages, $data) { 174*aa11d6a4Sleibler $pages = array(); 175*aa11d6a4Sleibler foreach($todopages as $page) { 176*aa11d6a4Sleibler $todos = array(); 177*aa11d6a4Sleibler // contains 3 arrays: an array with complete matches and 2 arrays with subpatterns 178*aa11d6a4Sleibler foreach($page['matches'][1] as $todoindex => $todomatch) { 179*aa11d6a4Sleibler list($checked, $todouser) = $this->parseTodoArgs($todomatch); 180*aa11d6a4Sleibler $todotitle = trim($page['matches'][2][$todoindex]); 181*aa11d6a4Sleibler 182*aa11d6a4Sleibler if($this->isRequestedTodo($data, $checked, $todouser)) { 183*aa11d6a4Sleibler $todos[] = array($todotitle, $todoindex, $todouser, $checked); 184*aa11d6a4Sleibler } 185*aa11d6a4Sleibler } 186*aa11d6a4Sleibler if(count($todos) > 0) { 187*aa11d6a4Sleibler $pages[] = array('id' => $page['id'], 'todos' => $todos); 188*aa11d6a4Sleibler } 189*aa11d6a4Sleibler } 190*aa11d6a4Sleibler return $pages; 191*aa11d6a4Sleibler } 192*aa11d6a4Sleibler 193*aa11d6a4Sleibler /** 194*aa11d6a4Sleibler * Create html for table with todos 195*aa11d6a4Sleibler * 196*aa11d6a4Sleibler * @param Doku_Renderer_xhtml $R 197*aa11d6a4Sleibler * @param array $todopages 198*aa11d6a4Sleibler */ 199*aa11d6a4Sleibler private function htmlTodoTable($R, $todopages) { 200*aa11d6a4Sleibler $R->table_open(); 201*aa11d6a4Sleibler foreach($todopages as $page) { 202*aa11d6a4Sleibler $R->tablerow_open(); 203*aa11d6a4Sleibler $R->tableheader_open(); 204*aa11d6a4Sleibler $R->internallink($page['id'], $page['id']); 205*aa11d6a4Sleibler $R->tableheader_close(); 206*aa11d6a4Sleibler $R->tablerow_close(); 207*aa11d6a4Sleibler foreach($page['todos'] as $todo) { 208*aa11d6a4Sleibler $R->tablerow_open(); 209*aa11d6a4Sleibler $R->tablecell_open(); 210*aa11d6a4Sleibler $R->doc .= $this->createTodoItem($R, $todo[0], $todo[1], $todo[2], $todo[3], $page['id']); 211*aa11d6a4Sleibler $R->tablecell_close(); 212*aa11d6a4Sleibler $R->tablerow_close(); 213*aa11d6a4Sleibler } 214*aa11d6a4Sleibler } 215*aa11d6a4Sleibler $R->table_close(); 216*aa11d6a4Sleibler } 217*aa11d6a4Sleibler 218*aa11d6a4Sleibler /** 219*aa11d6a4Sleibler * Check the conditions for adding a todoitem 220*aa11d6a4Sleibler * 221*aa11d6a4Sleibler * @param $data array the defined filters 222*aa11d6a4Sleibler * @param $checked bool completion status of task; true: finished, false: open 223*aa11d6a4Sleibler * @param $todouser string user username of user 224*aa11d6a4Sleibler * @return bool if the todoitem should be listed 225*aa11d6a4Sleibler */ 226*aa11d6a4Sleibler private function isRequestedTodo($data, $checked, $todouser) { 227*aa11d6a4Sleibler //completion status 228*aa11d6a4Sleibler $condition1 = $data['completed'] === 'all' //all 229*aa11d6a4Sleibler || $data['completed'] === $checked; //yes or no 230*aa11d6a4Sleibler 231*aa11d6a4Sleibler // resolve placeholder in assignees 232*aa11d6a4Sleibler $requestedassignees = array(); 233*aa11d6a4Sleibler if(is_array($data['assigned'])) { 234*aa11d6a4Sleibler $requestedassignees = array_map( 235*aa11d6a4Sleibler function($user) { 236*aa11d6a4Sleibler if($user == '@@USER@@' && !empty($_SERVER['REMOTE_USER'])) { //$INPUT->server->str('REMOTE_USER') 237*aa11d6a4Sleibler return $_SERVER['REMOTE_USER']; 238*aa11d6a4Sleibler } 239*aa11d6a4Sleibler return $user; 240*aa11d6a4Sleibler }, 241*aa11d6a4Sleibler $data['assigned'] 242*aa11d6a4Sleibler ); 243*aa11d6a4Sleibler } 244*aa11d6a4Sleibler //assigned 245*aa11d6a4Sleibler $condition2 = $data['assigned'] === 'all' //all 246*aa11d6a4Sleibler || (is_bool($data['assigned']) && $data['assigned'] == $todouser) //yes or no 247*aa11d6a4Sleibler || (is_array($data['assigned']) && in_array($todouser, $requestedassignees)); //one of the requested users? 248*aa11d6a4Sleibler 249*aa11d6a4Sleibler return $condition1 AND $condition2; 250*aa11d6a4Sleibler } 251*aa11d6a4Sleibler} 252