xref: /plugin/discussion/helper.php (revision 06ed893a2308b880b9a5d0cc2b0bdfb1088ac20a)
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&nbsp;'.$this->getLang('nocomments');
864cded5e1SGerrit Uitslag        } elseif ($num == 1) {
874cded5e1SGerrit Uitslag            $comment = '1&nbsp;'.$this->getLang('comment');
884cded5e1SGerrit Uitslag        } else {
894cded5e1SGerrit Uitslag            $comment = $num.'&nbsp;'.$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