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); 59aa11d6a4Sleibler $data = array( 60aa11d6a4Sleibler 'completed' => 'all', 61aa11d6a4Sleibler 'assigned' => 'all' 62aa11d6a4Sleibler ); 63aa11d6a4Sleibler $allowedvalues = array('yes', 'no'); 64aa11d6a4Sleibler foreach($options as $option) { 65aa11d6a4Sleibler @list($key, $value) = explode(':', $option, 2); 66aa11d6a4Sleibler switch($key) { 67aa11d6a4Sleibler case 'completed': 68aa11d6a4Sleibler if(in_array($value, $allowedvalues)) { 69aa11d6a4Sleibler $data['completed'] = ($value == 'yes'); 70aa11d6a4Sleibler } 71aa11d6a4Sleibler break; 72aa11d6a4Sleibler case 'assigned': 73aa11d6a4Sleibler if(in_array($value, $allowedvalues)) { 74aa11d6a4Sleibler $data['assigned'] = ($value == 'yes'); 75aa11d6a4Sleibler break; 76aa11d6a4Sleibler } 77aa11d6a4Sleibler //assigned? 78aa11d6a4Sleibler $data['assigned'] = explode(',', $value); 79*e0ec7364Sleibler // @date 20140317 le: if check for logged in user, also check for logged in user email address 80*e0ec7364Sleibler if( in_array( '@@USER@@', $data['assigned'] ) ) { 81*e0ec7364Sleibler $data['assigned'][] = '@@USER-MAIL@@'; 82*e0ec7364Sleibler } 83aa11d6a4Sleibler $data['assigned'] = array_map( 84aa11d6a4Sleibler function ($user) { 85aa11d6a4Sleibler //placeholder 86*e0ec7364Sleibler if( $user == '@@USER@@' || $user == '@@USER-MAIL@@' ) { 87aa11d6a4Sleibler return $user; 88aa11d6a4Sleibler } 89aa11d6a4Sleibler //user 90aa11d6a4Sleibler return ltrim($user, '@'); 91aa11d6a4Sleibler }, $data['assigned'] 92aa11d6a4Sleibler ); 93aa11d6a4Sleibler break; 94aa11d6a4Sleibler } 95aa11d6a4Sleibler } 96aa11d6a4Sleibler return $data; 97aa11d6a4Sleibler } 98aa11d6a4Sleibler 99aa11d6a4Sleibler /** 100aa11d6a4Sleibler * Render xhtml output or metadata 101aa11d6a4Sleibler * 102aa11d6a4Sleibler * @param string $mode Renderer mode (supported modes: xhtml) 103aa11d6a4Sleibler * @param Doku_Renderer $renderer The renderer 104aa11d6a4Sleibler * @param array $data The data from the handler() function 105aa11d6a4Sleibler * @return bool If rendering was successful. 106aa11d6a4Sleibler */ 107aa11d6a4Sleibler public function render($mode, Doku_Renderer &$renderer, $data) { 108aa11d6a4Sleibler global $conf; 109aa11d6a4Sleibler 110aa11d6a4Sleibler if($mode != 'xhtml') return false; 111aa11d6a4Sleibler /** @var Doku_Renderer_xhtml $renderer */ 112aa11d6a4Sleibler 113aa11d6a4Sleibler $opts['pattern'] = '/<todo([^>]*)>(.*)<\/todo[\W]*?>/'; //all todos in a wiki page 114aa11d6a4Sleibler //TODO check if storing subpatterns doesn't cost too much resources 115aa11d6a4Sleibler 116aa11d6a4Sleibler // search(&$data, $base, $func, $opts,$dir='',$lvl=1,$sort='natural') 117aa11d6a4Sleibler search($todopages, $conf['datadir'], array($this, 'search_todos'), $opts); //browse wiki pages with callback to search_pattern 118aa11d6a4Sleibler 119aa11d6a4Sleibler $todopages = $this->filterpages($todopages, $data); 120aa11d6a4Sleibler 121aa11d6a4Sleibler $this->htmlTodoTable($renderer, $todopages); 122aa11d6a4Sleibler 123aa11d6a4Sleibler return true; 124aa11d6a4Sleibler } 125aa11d6a4Sleibler 126aa11d6a4Sleibler /** 127aa11d6a4Sleibler * Custom search callback 128aa11d6a4Sleibler * 129aa11d6a4Sleibler * This function is called for every found file or 130aa11d6a4Sleibler * directory. When a directory is given to the function it has to 131aa11d6a4Sleibler * decide if this directory should be traversed (true) or not (false). 132aa11d6a4Sleibler * Return values for files are ignored 133aa11d6a4Sleibler * 134aa11d6a4Sleibler * All functions should check the ACL for document READ rights 135aa11d6a4Sleibler * namespaces (directories) are NOT checked (when sneaky_index is 0) as this 136aa11d6a4Sleibler * would break the recursion (You can have an nonreadable dir over a readable 137aa11d6a4Sleibler * one deeper nested) also make sure to check the file type (for example 138aa11d6a4Sleibler * in case of lockfiles). 139aa11d6a4Sleibler * 140aa11d6a4Sleibler * @param array &$data - Reference to the result data structure 141aa11d6a4Sleibler * @param string $base - Base usually $conf['datadir'] 142aa11d6a4Sleibler * @param string $file - current file or directory relative to $base 143aa11d6a4Sleibler * @param string $type - Type either 'd' for directory or 'f' for file 144aa11d6a4Sleibler * @param int $lvl - Current recursion depht 145aa11d6a4Sleibler * @param array $opts - option array as given to search() 146aa11d6a4Sleibler * @return bool if this directory should be traversed (true) or not (false). Return values for files are ignored. 147aa11d6a4Sleibler */ 148aa11d6a4Sleibler public function search_todos(&$data, $base, $file, $type, $lvl, $opts) { 149aa11d6a4Sleibler $item['id'] = pathID($file); //get current file ID 150aa11d6a4Sleibler 151aa11d6a4Sleibler //we do nothing with directories 152aa11d6a4Sleibler if($type == 'd') return true; 153aa11d6a4Sleibler 154aa11d6a4Sleibler //only search txt files 155aa11d6a4Sleibler if(substr($file, -4) != '.txt') return true; 156aa11d6a4Sleibler 157aa11d6a4Sleibler //check ACL 158aa11d6a4Sleibler if(auth_quickaclcheck($item['id']) < AUTH_READ) return false; 159aa11d6a4Sleibler 160aa11d6a4Sleibler $wikitext = rawWiki($item['id']); //get wiki text 161aa11d6a4Sleibler 162aa11d6a4Sleibler $item['count'] = preg_match_all($opts['pattern'], $wikitext, $matches); //count how many times appears the pattern 163aa11d6a4Sleibler if(!empty($item['count'])) { //if it appears at least once 164aa11d6a4Sleibler $item['matches'] = $matches; 165aa11d6a4Sleibler $data[] = $item; 166aa11d6a4Sleibler } 167aa11d6a4Sleibler return true; 168aa11d6a4Sleibler } 169aa11d6a4Sleibler 170aa11d6a4Sleibler /** 171aa11d6a4Sleibler * filter the pages 172aa11d6a4Sleibler * 173aa11d6a4Sleibler * @param $todopages array pages with all todoitems 174aa11d6a4Sleibler * @param $data array listing parameters 175aa11d6a4Sleibler * @return array filtered pages 176aa11d6a4Sleibler */ 177aa11d6a4Sleibler private function filterpages($todopages, $data) { 178aa11d6a4Sleibler $pages = array(); 179aa11d6a4Sleibler foreach($todopages as $page) { 180aa11d6a4Sleibler $todos = array(); 181aa11d6a4Sleibler // contains 3 arrays: an array with complete matches and 2 arrays with subpatterns 182aa11d6a4Sleibler foreach($page['matches'][1] as $todoindex => $todomatch) { 183aa11d6a4Sleibler list($checked, $todouser) = $this->parseTodoArgs($todomatch); 184aa11d6a4Sleibler $todotitle = trim($page['matches'][2][$todoindex]); 185aa11d6a4Sleibler 186aa11d6a4Sleibler if($this->isRequestedTodo($data, $checked, $todouser)) { 187aa11d6a4Sleibler $todos[] = array($todotitle, $todoindex, $todouser, $checked); 188aa11d6a4Sleibler } 189aa11d6a4Sleibler } 190aa11d6a4Sleibler if(count($todos) > 0) { 191aa11d6a4Sleibler $pages[] = array('id' => $page['id'], 'todos' => $todos); 192aa11d6a4Sleibler } 193aa11d6a4Sleibler } 194aa11d6a4Sleibler return $pages; 195aa11d6a4Sleibler } 196aa11d6a4Sleibler 197aa11d6a4Sleibler /** 198aa11d6a4Sleibler * Create html for table with todos 199aa11d6a4Sleibler * 200aa11d6a4Sleibler * @param Doku_Renderer_xhtml $R 201aa11d6a4Sleibler * @param array $todopages 202aa11d6a4Sleibler */ 203aa11d6a4Sleibler private function htmlTodoTable($R, $todopages) { 204aa11d6a4Sleibler $R->table_open(); 205aa11d6a4Sleibler foreach($todopages as $page) { 206aa11d6a4Sleibler $R->tablerow_open(); 207aa11d6a4Sleibler $R->tableheader_open(); 208aa11d6a4Sleibler $R->internallink($page['id'], $page['id']); 209aa11d6a4Sleibler $R->tableheader_close(); 210aa11d6a4Sleibler $R->tablerow_close(); 211aa11d6a4Sleibler foreach($page['todos'] as $todo) { 212aa11d6a4Sleibler $R->tablerow_open(); 213aa11d6a4Sleibler $R->tablecell_open(); 214aa11d6a4Sleibler $R->doc .= $this->createTodoItem($R, $todo[0], $todo[1], $todo[2], $todo[3], $page['id']); 215aa11d6a4Sleibler $R->tablecell_close(); 216aa11d6a4Sleibler $R->tablerow_close(); 217aa11d6a4Sleibler } 218aa11d6a4Sleibler } 219aa11d6a4Sleibler $R->table_close(); 220aa11d6a4Sleibler } 221aa11d6a4Sleibler 222aa11d6a4Sleibler /** 223aa11d6a4Sleibler * Check the conditions for adding a todoitem 224aa11d6a4Sleibler * 225aa11d6a4Sleibler * @param $data array the defined filters 226aa11d6a4Sleibler * @param $checked bool completion status of task; true: finished, false: open 227aa11d6a4Sleibler * @param $todouser string user username of user 228aa11d6a4Sleibler * @return bool if the todoitem should be listed 229aa11d6a4Sleibler */ 230aa11d6a4Sleibler private function isRequestedTodo($data, $checked, $todouser) { 231*e0ec7364Sleibler 232aa11d6a4Sleibler //completion status 233aa11d6a4Sleibler $condition1 = $data['completed'] === 'all' //all 234aa11d6a4Sleibler || $data['completed'] === $checked; //yes or no 235aa11d6a4Sleibler 236aa11d6a4Sleibler // resolve placeholder in assignees 237aa11d6a4Sleibler $requestedassignees = array(); 238aa11d6a4Sleibler if(is_array($data['assigned'])) { 239aa11d6a4Sleibler $requestedassignees = array_map( 240aa11d6a4Sleibler function($user) { 241*e0ec7364Sleibler global $USERINFO; 242aa11d6a4Sleibler if($user == '@@USER@@' && !empty($_SERVER['REMOTE_USER'])) { //$INPUT->server->str('REMOTE_USER') 243aa11d6a4Sleibler return $_SERVER['REMOTE_USER']; 244aa11d6a4Sleibler } 245*e0ec7364Sleibler // @date 20140317 le: check for logged in user email address 246*e0ec7364Sleibler if( $user == '@@USER-MAIL@@' && isset( $USERINFO['mail'] ) ) { 247*e0ec7364Sleibler return $USERINFO['mail']; 248*e0ec7364Sleibler } 249aa11d6a4Sleibler return $user; 250aa11d6a4Sleibler }, 251aa11d6a4Sleibler $data['assigned'] 252aa11d6a4Sleibler ); 253aa11d6a4Sleibler } 254aa11d6a4Sleibler //assigned 255aa11d6a4Sleibler $condition2 = $data['assigned'] === 'all' //all 256aa11d6a4Sleibler || (is_bool($data['assigned']) && $data['assigned'] == $todouser) //yes or no 257aa11d6a4Sleibler || (is_array($data['assigned']) && in_array($todouser, $requestedassignees)); //one of the requested users? 258aa11d6a4Sleibler 259aa11d6a4Sleibler return $condition1 AND $condition2; 260aa11d6a4Sleibler } 261aa11d6a4Sleibler} 262