xref: /plugin/discussion/helper.php (revision 55e1d1446c1364106aee6fd742edf15c32d08c27)
15fc512fbSwikidesign<?php
25fc512fbSwikidesign/**
35fc512fbSwikidesign * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
45fc512fbSwikidesign * @author     Esther Brunner <wikidesign@gmail.com>
55fc512fbSwikidesign */
65fc512fbSwikidesign
75fc512fbSwikidesign// must be run within Dokuwiki
85fc512fbSwikidesignif (!defined('DOKU_INC')) die();
95fc512fbSwikidesign
105fc512fbSwikidesignif (!defined('DOKU_LF')) define('DOKU_LF', "\n");
115fc512fbSwikidesignif (!defined('DOKU_TAB')) define('DOKU_TAB', "\t");
125fc512fbSwikidesign
135fc512fbSwikidesignclass helper_plugin_discussion extends DokuWiki_Plugin {
145fc512fbSwikidesign
155fc512fbSwikidesign    function getInfo() {
165fc512fbSwikidesign        return array(
174c4d91c6SGina Haeussge                'author' => 'Gina Häußge, Michael Klier, Esther Brunner',
184c4d91c6SGina Haeussge                'email'  => 'dokuwiki@chimeric.de',
193648fb3eSMichael Klier                'date'   => @file_get_contents(DOKU_PLUGIN.'discussion/VERSION'),
205fc512fbSwikidesign                'name'   => 'Discussion Plugin (helper class)',
215fc512fbSwikidesign                'desc'   => 'Functions to get info about comments to a wiki page',
224c4d91c6SGina Haeussge                'url'    => 'http://wiki.splitbrain.org/plugin:discussion',
235fc512fbSwikidesign                );
245fc512fbSwikidesign    }
255fc512fbSwikidesign
265fc512fbSwikidesign    function getMethods() {
275fc512fbSwikidesign        $result = array();
285fc512fbSwikidesign        $result[] = array(
295fc512fbSwikidesign                'name'   => 'th',
305fc512fbSwikidesign                'desc'   => 'returns the header of the comments column for pagelist',
315fc512fbSwikidesign                'return' => array('header' => 'string'),
325fc512fbSwikidesign                );
335fc512fbSwikidesign        $result[] = array(
345fc512fbSwikidesign                'name'   => 'td',
355fc512fbSwikidesign                'desc'   => 'returns the link to the discussion section with number of comments',
365fc512fbSwikidesign                'params' => array(
375fc512fbSwikidesign                    'id' => 'string',
385fc512fbSwikidesign                    'number of comments (optional)' => 'integer'),
395fc512fbSwikidesign                'return' => array('link' => 'string'),
405fc512fbSwikidesign                );
415fc512fbSwikidesign        $result[] = array(
425fc512fbSwikidesign                'name'   => 'getThreads',
435fc512fbSwikidesign                'desc'   => 'returns pages with discussion sections, sorted by recent comments',
445fc512fbSwikidesign                'params' => array(
455fc512fbSwikidesign                    'namespace' => 'string',
465fc512fbSwikidesign                    'number (optional)' => 'integer'),
475fc512fbSwikidesign                'return' => array('pages' => 'array'),
485fc512fbSwikidesign                );
495fc512fbSwikidesign        $result[] = array(
505fc512fbSwikidesign                'name'   => 'getComments',
515fc512fbSwikidesign                'desc'   => 'returns recently added or edited comments individually',
525fc512fbSwikidesign                'params' => array(
535fc512fbSwikidesign                    'namespace' => 'string',
545fc512fbSwikidesign                    'number (optional)' => 'integer'),
555fc512fbSwikidesign                'return' => array('pages' => 'array'),
565fc512fbSwikidesign                );
575fc512fbSwikidesign        return $result;
585fc512fbSwikidesign    }
595fc512fbSwikidesign
605fc512fbSwikidesign    /**
615fc512fbSwikidesign     * Returns the column header for the Pagelist Plugin
625fc512fbSwikidesign     */
635fc512fbSwikidesign    function th() {
645fc512fbSwikidesign        return $this->getLang('discussion');
655fc512fbSwikidesign    }
665fc512fbSwikidesign
675fc512fbSwikidesign    /**
685fc512fbSwikidesign     * Returns the link to the discussion section of a page
695fc512fbSwikidesign     */
705fc512fbSwikidesign    function td($id, $num = NULL) {
71479dd10fSwikidesign        $section = '#discussion__section';
725fc512fbSwikidesign
735fc512fbSwikidesign        if (!isset($num)) {
745fc512fbSwikidesign            $cfile = metaFN($id, '.comments');
755fc512fbSwikidesign            $comments = unserialize(io_readFile($cfile, false));
765fc512fbSwikidesign
775fc512fbSwikidesign            $num = $comments['number'];
785fc512fbSwikidesign            if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return '';
795fc512fbSwikidesign        }
805fc512fbSwikidesign
81077a3d86SGina Haeussge        if ($num == 0) $comment = '0&nbsp;'.$this->getLang($this->getLang('nocomments'));
82fbb94835Swikidesign        elseif ($num == 1) $comment = '1&nbsp;'.$this->getLang('comment');
83fbb94835Swikidesign        else $comment = $num.'&nbsp;'.$this->getLang('comments');
845fc512fbSwikidesign
85479dd10fSwikidesign        return '<a href="'.wl($id).$section.'" class="wikilink1" title="'.$id.$section.'">'.
865fc512fbSwikidesign            $comment.'</a>';
875fc512fbSwikidesign    }
885fc512fbSwikidesign
895fc512fbSwikidesign    /**
905fc512fbSwikidesign     * Returns an array of pages with discussion sections, sorted by recent comments
915fc512fbSwikidesign     */
92573e23a1Swikidesign    function getThreads($ns, $num = NULL) {
935fc512fbSwikidesign        global $conf;
945fc512fbSwikidesign
955fc512fbSwikidesign        require_once(DOKU_INC.'inc/search.php');
965fc512fbSwikidesign
975fc512fbSwikidesign        $dir = $conf['datadir'].($ns ? '/'.str_replace(':', '/', $ns): '');
985fc512fbSwikidesign
995fc512fbSwikidesign        // returns the list of pages in the given namespace and it's subspaces
1005fc512fbSwikidesign        $items = array();
1015fc512fbSwikidesign        search($items, $dir, 'search_allpages', '');
1025fc512fbSwikidesign
1035fc512fbSwikidesign        // add pages with comments to result
1045fc512fbSwikidesign        $result = array();
1055fc512fbSwikidesign        foreach ($items as $item) {
1065fc512fbSwikidesign            $id   = ($ns ? $ns.':' : '').$item['id'];
1075fc512fbSwikidesign
1085fc512fbSwikidesign            // some checks
1095fc512fbSwikidesign            $perm = auth_quickaclcheck($id);
1105fc512fbSwikidesign            if ($perm < AUTH_READ) continue;    // skip if no permission
1115fc512fbSwikidesign            $file = metaFN($id, '.comments');
1125fc512fbSwikidesign            if (!@file_exists($file)) continue; // skip if no comments file
1135fc512fbSwikidesign            $data = unserialize(io_readFile($file, false));
114264b7327Swikidesign            $status = $data['status'];
115fcb1bc77Swikidesign            $number = $data['number']; // skip if comments are off or closed without comments
116573e23a1Swikidesign            if (!$status || (($status == 2) && (!$number))) continue;
1175fc512fbSwikidesign
1185fc512fbSwikidesign            $date = filemtime($file);
1195fc512fbSwikidesign            $meta = p_get_metadata($id);
1205fc512fbSwikidesign            $result[$date] = array(
1215fc512fbSwikidesign                    'id'       => $id,
122264b7327Swikidesign                    'file'     => $file,
1235fc512fbSwikidesign                    'title'    => $meta['title'],
1245fc512fbSwikidesign                    'date'     => $date,
1255fc512fbSwikidesign                    'user'     => $meta['creator'],
1265fc512fbSwikidesign                    'desc'     => $meta['description']['abstract'],
127fcb1bc77Swikidesign                    'num'      => $number,
128fcb1bc77Swikidesign                    'comments' => $this->td($id, $number),
129264b7327Swikidesign                    'status'   => $status,
1305fc512fbSwikidesign                    'perm'     => $perm,
1315fc512fbSwikidesign                    'exists'   => true,
13273f66a3cSwikidesign                    'anchor'   => 'discussion__section',
1335fc512fbSwikidesign                    );
1345fc512fbSwikidesign        }
1355fc512fbSwikidesign
1365fc512fbSwikidesign        // finally sort by time of last comment
1375fc512fbSwikidesign        krsort($result);
1385fc512fbSwikidesign
1395fc512fbSwikidesign        if (is_numeric($num)) $result = array_slice($result, 0, $num);
1405fc512fbSwikidesign
1415fc512fbSwikidesign        return $result;
1425fc512fbSwikidesign    }
1435fc512fbSwikidesign
1445fc512fbSwikidesign    /**
1455fc512fbSwikidesign     * Returns an array of recently added comments to a given page or namespace
1465fc512fbSwikidesign     */
1475fc512fbSwikidesign    function getComments($ns, $num = NULL) {
1485fc512fbSwikidesign        global $conf;
1495fc512fbSwikidesign
1505fc512fbSwikidesign        $first  = $_REQUEST['first'];
1515fc512fbSwikidesign        if (!is_numeric($first)) $first = 0;
1525fc512fbSwikidesign
1535fc512fbSwikidesign        if ((!$num) || (!is_numeric($num))) $num = $conf['recent'];
1545fc512fbSwikidesign
1555fc512fbSwikidesign        $result = array();
1565fc512fbSwikidesign        $count  = 0;
1575fc512fbSwikidesign
1585fc512fbSwikidesign        if (!@file_exists($conf['metadir'].'/_comments.changes')) return $result;
1595fc512fbSwikidesign
1605fc512fbSwikidesign        // read all recent changes. (kept short)
1615fc512fbSwikidesign        $lines = file($conf['metadir'].'/_comments.changes');
1625fc512fbSwikidesign
163*55e1d144SMichael Hamann        $seen = array(); //caches seen pages in order to skip them
1645fc512fbSwikidesign        // handle lines
165*55e1d144SMichael Hamann        $line_num = count($lines);
166*55e1d144SMichael Hamann        for ($i = ($line_num - 1); $i >= 0; $i--) {
167*55e1d144SMichael Hamann            $rec = $this->_handleRecentComment($lines[$i], $ns, $seen);
1685fc512fbSwikidesign            if ($rec !== false) {
1695fc512fbSwikidesign                if (--$first >= 0) continue; // skip first entries
1705fc512fbSwikidesign                $result[$rec['date']] = $rec;
1715fc512fbSwikidesign                $count++;
1725fc512fbSwikidesign                // break when we have enough entries
1735fc512fbSwikidesign                if ($count >= $num) break;
1745fc512fbSwikidesign            }
1755fc512fbSwikidesign        }
1765fc512fbSwikidesign
1775fc512fbSwikidesign        // finally sort by time of last comment
1785fc512fbSwikidesign        krsort($result);
1795fc512fbSwikidesign
1805fc512fbSwikidesign        return $result;
1815fc512fbSwikidesign    }
1825fc512fbSwikidesign
1835fc512fbSwikidesign    /* ---------- Changelog function adapted for the Discussion Plugin ---------- */
1845fc512fbSwikidesign
1855fc512fbSwikidesign    /**
1865fc512fbSwikidesign     * Internal function used by $this->getComments()
1875fc512fbSwikidesign     *
1885fc512fbSwikidesign     * don't call directly
1895fc512fbSwikidesign     *
1905fc512fbSwikidesign     * @see getRecentComments()
1915fc512fbSwikidesign     * @author Andreas Gohr <andi@splitbrain.org>
1925fc512fbSwikidesign     * @author Ben Coburn <btcoburn@silicodon.net>
1935fc512fbSwikidesign     * @author Esther Brunner <wikidesign@gmail.com>
1945fc512fbSwikidesign     */
195*55e1d144SMichael Hamann    function _handleRecentComment($line, $ns, &$seen) {
1965fc512fbSwikidesign        if (empty($line)) return false;  //skip empty lines
1975fc512fbSwikidesign
1985fc512fbSwikidesign        // split the line into parts
1995fc512fbSwikidesign        $recent = parseChangelogLine($line);
2005fc512fbSwikidesign        if ($recent === false) return false;
2015fc512fbSwikidesign
2025fc512fbSwikidesign        $cid     = $recent['extra'];
2035fc512fbSwikidesign        $fullcid = $recent['id'].'#'.$recent['extra'];
2045fc512fbSwikidesign
2055fc512fbSwikidesign        // skip seen ones
2065fc512fbSwikidesign        if (isset($seen[$fullcid])) return false;
2075fc512fbSwikidesign
2085fc512fbSwikidesign        // skip 'show comment' log entries
2095fc512fbSwikidesign        if ($recent['type'] === 'sc') return false;
2105fc512fbSwikidesign
2115fc512fbSwikidesign        // remember in seen to skip additional sights
2125fc512fbSwikidesign        $seen[$fullcid] = 1;
2135fc512fbSwikidesign
2145fc512fbSwikidesign        // check if it's a hidden page or comment
2155fc512fbSwikidesign        if (isHiddenPage($recent['id'])) return false;
2165fc512fbSwikidesign        if ($recent['type'] === 'hc') return false;
2175fc512fbSwikidesign
2185fc512fbSwikidesign        // filter namespace or id
2195fc512fbSwikidesign        if (($ns) && (strpos($recent['id'].':', $ns.':') !== 0)) return false;
2205fc512fbSwikidesign
2215fc512fbSwikidesign        // check ACL
2225fc512fbSwikidesign        $recent['perm'] = auth_quickaclcheck($recent['id']);
2235fc512fbSwikidesign        if ($recent['perm'] < AUTH_READ) return false;
2245fc512fbSwikidesign
2255fc512fbSwikidesign        // check existance
2265fc512fbSwikidesign        $recent['file'] = wikiFN($recent['id']);
2275fc512fbSwikidesign        $recent['exists'] = @file_exists($recent['file']);
2285fc512fbSwikidesign        if (!$recent['exists']) return false;
2295fc512fbSwikidesign        if ($recent['type'] === 'dc') return false;
2305fc512fbSwikidesign
2315fc512fbSwikidesign        // get discussion meta file name
2322cbdac2eSwikidesign        $data = unserialize(io_readFile(metaFN($recent['id'], '.comments'), false));
2335fc512fbSwikidesign
2345fc512fbSwikidesign        // check if discussion is turned off
2355fc512fbSwikidesign        if ($data['status'] === 0) return false;
2365fc512fbSwikidesign
237*55e1d144SMichael Hamann        // check if the comment still exists
238*55e1d144SMichael Hamann        if (!isset($data['comments'][$cid])) return false;
239*55e1d144SMichael Hamann
2405fc512fbSwikidesign        // okay, then add some additional info
2416046f25cSwikidesign        if (is_array($data['comments'][$cid]['user']))
2426046f25cSwikidesign            $recent['name'] = $data['comments'][$cid]['user']['name'];
2436046f25cSwikidesign        else $recent['name'] = $data['comments'][$cid]['name'];
2445fc512fbSwikidesign        $recent['desc'] = strip_tags($data['comments'][$cid]['xhtml']);
245402011e1SGina Haeussge        $recent['anchor'] = 'comment__'.$cid;
2465fc512fbSwikidesign
2475fc512fbSwikidesign        return $recent;
2485fc512fbSwikidesign    }
2495fc512fbSwikidesign}
250530693fbSMichael Klier// vim:ts=4:sw=4:et:enc=utf-8:
251