15fc512fbSwikidesign<?php 25fc512fbSwikidesign/** 35fc512fbSwikidesign * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 45fc512fbSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 55fc512fbSwikidesign */ 65fc512fbSwikidesign 7e7ac9adaSGerrit Uitslag/** 8e7ac9adaSGerrit Uitslag * Class helper_plugin_discussion 9e7ac9adaSGerrit Uitslag */ 105fc512fbSwikidesignclass helper_plugin_discussion extends DokuWiki_Plugin { 115fc512fbSwikidesign 12e7ac9adaSGerrit Uitslag /** 13e7ac9adaSGerrit Uitslag * @return array 14e7ac9adaSGerrit Uitslag */ 15f3535bedSGerrit Uitslag public function getMethods() { 165fc512fbSwikidesign $result = array(); 175fc512fbSwikidesign $result[] = array( 185fc512fbSwikidesign 'name' => 'th', 195fc512fbSwikidesign 'desc' => 'returns the header of the comments column for pagelist', 205fc512fbSwikidesign 'return' => array('header' => 'string'), 215fc512fbSwikidesign ); 225fc512fbSwikidesign $result[] = array( 235fc512fbSwikidesign 'name' => 'td', 245fc512fbSwikidesign 'desc' => 'returns the link to the discussion section with number of comments', 255fc512fbSwikidesign 'params' => array( 265fc512fbSwikidesign 'id' => 'string', 275fc512fbSwikidesign 'number of comments (optional)' => 'integer'), 285fc512fbSwikidesign 'return' => array('link' => 'string'), 295fc512fbSwikidesign ); 305fc512fbSwikidesign $result[] = array( 315fc512fbSwikidesign 'name' => 'getThreads', 325fc512fbSwikidesign 'desc' => 'returns pages with discussion sections, sorted by recent comments', 335fc512fbSwikidesign 'params' => array( 345fc512fbSwikidesign 'namespace' => 'string', 355fc512fbSwikidesign 'number (optional)' => 'integer'), 365fc512fbSwikidesign 'return' => array('pages' => 'array'), 375fc512fbSwikidesign ); 385fc512fbSwikidesign $result[] = array( 395fc512fbSwikidesign 'name' => 'getComments', 405fc512fbSwikidesign 'desc' => 'returns recently added or edited comments individually', 415fc512fbSwikidesign 'params' => array( 425fc512fbSwikidesign 'namespace' => 'string', 435fc512fbSwikidesign 'number (optional)' => 'integer'), 445fc512fbSwikidesign 'return' => array('pages' => 'array'), 455fc512fbSwikidesign ); 46f3535bedSGerrit Uitslag $result[] = array( 47f3535bedSGerrit Uitslag 'name' => 'isDiscussionModerator', 48f3535bedSGerrit Uitslag 'desc' => 'check if current user is member of moderator groups', 49f3535bedSGerrit Uitslag 'params' => array(), 50f3535bedSGerrit Uitslag 'return' => array('isModerator' => 'boolean') 51f3535bedSGerrit Uitslag ); 525fc512fbSwikidesign return $result; 535fc512fbSwikidesign } 545fc512fbSwikidesign 555fc512fbSwikidesign /** 565fc512fbSwikidesign * Returns the column header for the Pagelist Plugin 57e7ac9adaSGerrit Uitslag * 58e7ac9adaSGerrit Uitslag * @return string 595fc512fbSwikidesign */ 60f3535bedSGerrit Uitslag public function th() { 615fc512fbSwikidesign return $this->getLang('discussion'); 625fc512fbSwikidesign } 635fc512fbSwikidesign 645fc512fbSwikidesign /** 655fc512fbSwikidesign * Returns the link to the discussion section of a page 66e7ac9adaSGerrit Uitslag * 67*06ed893aSGerrit Uitslag * @param string $id page id 68*06ed893aSGerrit Uitslag * @param string $col column name, used if more columns needed per plugin 69*06ed893aSGerrit Uitslag * @param string $class class name per cell set by reference 70*06ed893aSGerrit Uitslag * @param null|int $num number of visible comments -- internally used, not by pagelist plugin 71e7ac9adaSGerrit Uitslag * @return string 725fc512fbSwikidesign */ 73*06ed893aSGerrit Uitslag public function td($id, $col = null, &$class=null, $num = null) { 74479dd10fSwikidesign $section = '#discussion__section'; 755fc512fbSwikidesign 765fc512fbSwikidesign if (!isset($num)) { 775fc512fbSwikidesign $cfile = metaFN($id, '.comments'); 785fc512fbSwikidesign $comments = unserialize(io_readFile($cfile, false)); 795fc512fbSwikidesign 805fc512fbSwikidesign $num = $comments['number']; 815fc512fbSwikidesign if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return ''; 825fc512fbSwikidesign } 835fc512fbSwikidesign 844cded5e1SGerrit Uitslag if ($num == 0) { 854cded5e1SGerrit Uitslag $comment = '0 '.$this->getLang('nocomments'); 864cded5e1SGerrit Uitslag } elseif ($num == 1) { 874cded5e1SGerrit Uitslag $comment = '1 '.$this->getLang('comment'); 884cded5e1SGerrit Uitslag } else { 894cded5e1SGerrit Uitslag $comment = $num.' '.$this->getLang('comments'); 904cded5e1SGerrit Uitslag } 915fc512fbSwikidesign 92479dd10fSwikidesign return '<a href="'.wl($id).$section.'" class="wikilink1" title="'.$id.$section.'">'. 935fc512fbSwikidesign $comment.'</a>'; 945fc512fbSwikidesign } 955fc512fbSwikidesign 965fc512fbSwikidesign /** 975fc512fbSwikidesign * Returns an array of pages with discussion sections, sorted by recent comments 984cded5e1SGerrit Uitslag * Note: also used for content by Feed Plugin 99e7ac9adaSGerrit Uitslag * 100e7ac9adaSGerrit Uitslag * @param string $ns 101e7ac9adaSGerrit Uitslag * @param null|int $num 102e7ac9adaSGerrit Uitslag * @param string|bool $skipEmpty 103e7ac9adaSGerrit Uitslag * @return array 1045fc512fbSwikidesign */ 105f3535bedSGerrit Uitslag public function getThreads($ns, $num = null, $skipEmpty = false) { 1065fc512fbSwikidesign global $conf; 1075fc512fbSwikidesign 1085fc512fbSwikidesign require_once(DOKU_INC.'inc/search.php'); 1095fc512fbSwikidesign 11093bbea85SMichael Hamann $dir = $conf['datadir'].utf8_encodeFN(($ns ? '/'.str_replace(':', '/', $ns): '')); 1115fc512fbSwikidesign 1125fc512fbSwikidesign // returns the list of pages in the given namespace and it's subspaces 1135fc512fbSwikidesign $items = array(); 1145d76f934SMichael Hamann search($items, $dir, 'search_allpages', array()); 1155fc512fbSwikidesign 1165fc512fbSwikidesign // add pages with comments to result 1175fc512fbSwikidesign $result = array(); 1185fc512fbSwikidesign foreach ($items as $item) { 1195fc512fbSwikidesign $id = ($ns ? $ns.':' : '').$item['id']; 1205fc512fbSwikidesign 1215fc512fbSwikidesign // some checks 1225fc512fbSwikidesign $perm = auth_quickaclcheck($id); 1235fc512fbSwikidesign if ($perm < AUTH_READ) continue; // skip if no permission 1245fc512fbSwikidesign $file = metaFN($id, '.comments'); 1255fc512fbSwikidesign if (!@file_exists($file)) continue; // skip if no comments file 1265fc512fbSwikidesign $data = unserialize(io_readFile($file, false)); 127264b7327Swikidesign $status = $data['status']; 1285644a1afSlupo49 $number = $data['number']; 1295644a1afSlupo49 1305644a1afSlupo49 if (!$status || (($status == 2) && (!$number))) continue; // skip if comments are off or closed without comments 1315644a1afSlupo49 if($skipEmpty == 'y' && $number == 0) continue; // skip if discussion is empty and flag is set 1325fc512fbSwikidesign 1335fc512fbSwikidesign $date = filemtime($file); 1345fc512fbSwikidesign $meta = p_get_metadata($id); 135ae836b78SMichael Hamann $result[$date.'_'.$id] = array( 1365fc512fbSwikidesign 'id' => $id, 137264b7327Swikidesign 'file' => $file, 1385fc512fbSwikidesign 'title' => $meta['title'], 1395fc512fbSwikidesign 'date' => $date, 1405fc512fbSwikidesign 'user' => $meta['creator'], 1415fc512fbSwikidesign 'desc' => $meta['description']['abstract'], 142fcb1bc77Swikidesign 'num' => $number, 143*06ed893aSGerrit Uitslag 'comments' => $this->td($id, null, $class, $number), 144264b7327Swikidesign 'status' => $status, 1455fc512fbSwikidesign 'perm' => $perm, 1465fc512fbSwikidesign 'exists' => true, 14773f66a3cSwikidesign 'anchor' => 'discussion__section', 1485fc512fbSwikidesign ); 1495fc512fbSwikidesign } 1505fc512fbSwikidesign 1515fc512fbSwikidesign // finally sort by time of last comment 1525fc512fbSwikidesign krsort($result); 1535fc512fbSwikidesign 1545fc512fbSwikidesign if (is_numeric($num)) $result = array_slice($result, 0, $num); 1555fc512fbSwikidesign 1565fc512fbSwikidesign return $result; 1575fc512fbSwikidesign } 1585fc512fbSwikidesign 1595fc512fbSwikidesign /** 1605fc512fbSwikidesign * Returns an array of recently added comments to a given page or namespace 1614cded5e1SGerrit Uitslag * Note: also used for content by Feed Plugin 162e7ac9adaSGerrit Uitslag * 163e7ac9adaSGerrit Uitslag * @param string $ns 164e7ac9adaSGerrit Uitslag * @param int|null $num 165e7ac9adaSGerrit Uitslag * @return array 1665fc512fbSwikidesign */ 167f3535bedSGerrit Uitslag public function getComments($ns, $num = NULL) { 1685fc512fbSwikidesign global $conf; 1695fc512fbSwikidesign 1705fc512fbSwikidesign $first = $_REQUEST['first']; 1715fc512fbSwikidesign if (!is_numeric($first)) $first = 0; 1725fc512fbSwikidesign 1735fc512fbSwikidesign if ((!$num) || (!is_numeric($num))) $num = $conf['recent']; 1745fc512fbSwikidesign 1755fc512fbSwikidesign $result = array(); 1765fc512fbSwikidesign $count = 0; 1775fc512fbSwikidesign 1785fc512fbSwikidesign if (!@file_exists($conf['metadir'].'/_comments.changes')) return $result; 1795fc512fbSwikidesign 1805fc512fbSwikidesign // read all recent changes. (kept short) 1815fc512fbSwikidesign $lines = file($conf['metadir'].'/_comments.changes'); 1825fc512fbSwikidesign 18355e1d144SMichael Hamann $seen = array(); //caches seen pages in order to skip them 1845fc512fbSwikidesign // handle lines 18555e1d144SMichael Hamann $line_num = count($lines); 18655e1d144SMichael Hamann for ($i = ($line_num - 1); $i >= 0; $i--) { 18755e1d144SMichael Hamann $rec = $this->_handleRecentComment($lines[$i], $ns, $seen); 1885fc512fbSwikidesign if ($rec !== false) { 1895fc512fbSwikidesign if (--$first >= 0) continue; // skip first entries 1905fc512fbSwikidesign $result[$rec['date']] = $rec; 1915fc512fbSwikidesign $count++; 1925fc512fbSwikidesign // break when we have enough entries 1935fc512fbSwikidesign if ($count >= $num) break; 1945fc512fbSwikidesign } 1955fc512fbSwikidesign } 1965fc512fbSwikidesign 1975fc512fbSwikidesign // finally sort by time of last comment 1985fc512fbSwikidesign krsort($result); 1995fc512fbSwikidesign 2005fc512fbSwikidesign return $result; 2015fc512fbSwikidesign } 2025fc512fbSwikidesign 2035fc512fbSwikidesign /* ---------- Changelog function adapted for the Discussion Plugin ---------- */ 2045fc512fbSwikidesign 2055fc512fbSwikidesign /** 2065fc512fbSwikidesign * Internal function used by $this->getComments() 2075fc512fbSwikidesign * 2085fc512fbSwikidesign * don't call directly 2095fc512fbSwikidesign * 2105fc512fbSwikidesign * @see getRecentComments() 2115fc512fbSwikidesign * @author Andreas Gohr <andi@splitbrain.org> 2125fc512fbSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 2135fc512fbSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 214e7ac9adaSGerrit Uitslag * 215e7ac9adaSGerrit Uitslag * @param string $line 216e7ac9adaSGerrit Uitslag * @param string $ns 217e7ac9adaSGerrit Uitslag * @param array $seen 218e7ac9adaSGerrit Uitslag * @return array|bool 2195fc512fbSwikidesign */ 220f3535bedSGerrit Uitslag protected function _handleRecentComment($line, $ns, &$seen) { 2215fc512fbSwikidesign if (empty($line)) return false; //skip empty lines 2225fc512fbSwikidesign 2235fc512fbSwikidesign // split the line into parts 2245fc512fbSwikidesign $recent = parseChangelogLine($line); 2255fc512fbSwikidesign if ($recent === false) return false; 2265fc512fbSwikidesign 2275fc512fbSwikidesign $cid = $recent['extra']; 2285fc512fbSwikidesign $fullcid = $recent['id'].'#'.$recent['extra']; 2295fc512fbSwikidesign 2305fc512fbSwikidesign // skip seen ones 2315fc512fbSwikidesign if (isset($seen[$fullcid])) return false; 2325fc512fbSwikidesign 2335fc512fbSwikidesign // skip 'show comment' log entries 2345fc512fbSwikidesign if ($recent['type'] === 'sc') return false; 2355fc512fbSwikidesign 2365fc512fbSwikidesign // remember in seen to skip additional sights 2375fc512fbSwikidesign $seen[$fullcid] = 1; 2385fc512fbSwikidesign 2395fc512fbSwikidesign // check if it's a hidden page or comment 2405fc512fbSwikidesign if (isHiddenPage($recent['id'])) return false; 2415fc512fbSwikidesign if ($recent['type'] === 'hc') return false; 2425fc512fbSwikidesign 2435fc512fbSwikidesign // filter namespace or id 2445fc512fbSwikidesign if (($ns) && (strpos($recent['id'].':', $ns.':') !== 0)) return false; 2455fc512fbSwikidesign 2465fc512fbSwikidesign // check ACL 2475fc512fbSwikidesign $recent['perm'] = auth_quickaclcheck($recent['id']); 2485fc512fbSwikidesign if ($recent['perm'] < AUTH_READ) return false; 2495fc512fbSwikidesign 2505fc512fbSwikidesign // check existance 2515fc512fbSwikidesign $recent['file'] = wikiFN($recent['id']); 2525fc512fbSwikidesign $recent['exists'] = @file_exists($recent['file']); 2535fc512fbSwikidesign if (!$recent['exists']) return false; 2545fc512fbSwikidesign if ($recent['type'] === 'dc') return false; 2555fc512fbSwikidesign 2565fc512fbSwikidesign // get discussion meta file name 2572cbdac2eSwikidesign $data = unserialize(io_readFile(metaFN($recent['id'], '.comments'), false)); 2585fc512fbSwikidesign 2595fc512fbSwikidesign // check if discussion is turned off 2605fc512fbSwikidesign if ($data['status'] === 0) return false; 2615fc512fbSwikidesign 2624dd9b9e2SMichael Hamann $parent_id = $cid; 2634dd9b9e2SMichael Hamann // Check for the comment and all parents if they exist and are visible. 2644dd9b9e2SMichael Hamann do { 2654dd9b9e2SMichael Hamann $tcid = $parent_id; 2664dd9b9e2SMichael Hamann 26755e1d144SMichael Hamann // check if the comment still exists 2684dd9b9e2SMichael Hamann if (!isset($data['comments'][$tcid])) return false; 2694dd9b9e2SMichael Hamann // check if the comment is visible 2704dd9b9e2SMichael Hamann if ($data['comments'][$tcid]['show'] != 1) return false; 2714dd9b9e2SMichael Hamann 2724dd9b9e2SMichael Hamann $parent_id = $data['comments'][$tcid]['parent']; 2734dd9b9e2SMichael Hamann } while ($parent_id && $parent_id != $tcid); 27455e1d144SMichael Hamann 2755fc512fbSwikidesign // okay, then add some additional info 2764cded5e1SGerrit Uitslag if (is_array($data['comments'][$cid]['user'])) { 2776046f25cSwikidesign $recent['name'] = $data['comments'][$cid]['user']['name']; 2784cded5e1SGerrit Uitslag } else { 2794cded5e1SGerrit Uitslag $recent['name'] = $data['comments'][$cid]['name']; 2804cded5e1SGerrit Uitslag } 2815fc512fbSwikidesign $recent['desc'] = strip_tags($data['comments'][$cid]['xhtml']); 282d8092064SMichael Hamann $recent['anchor'] = 'comment_'.$cid; 2835fc512fbSwikidesign 2845fc512fbSwikidesign return $recent; 2855fc512fbSwikidesign } 286e6b2f142Slupo49 287e7ac9adaSGerrit Uitslag /** 288e7ac9adaSGerrit Uitslag * @return bool 289e7ac9adaSGerrit Uitslag */ 290f3535bedSGerrit Uitslag public function isDiscussionModerator() { 291e6b2f142Slupo49 global $USERINFO; 292e6b2f142Slupo49 $groups = trim($this->getConf('moderatorgroups')); 293e6b2f142Slupo49 294e6b2f142Slupo49 if(auth_ismanager()) return true; 295e6b2f142Slupo49 // Check if user is member of the moderator groups 296e6b2f142Slupo49 if(!empty($groups) && auth_isMember($groups, $_SERVER['REMOTE_USER'], (array)$USERINFO['grps'])) return true; 297e6b2f142Slupo49 298e6b2f142Slupo49 return false; 299e6b2f142Slupo49 } 3005fc512fbSwikidesign} 301530693fbSMichael Klier// vim:ts=4:sw=4:et:enc=utf-8: 302