xref: /plugin/discussion/action.php (revision 9881d83591eb8612b7a398091b9c8137d5a7af2b)
1f0fda08aSwikidesign<?php
2f0fda08aSwikidesign/**
3f0fda08aSwikidesign * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
4f0fda08aSwikidesign * @author     Esther Brunner <wikidesign@gmail.com>
5f0fda08aSwikidesign */
6f0fda08aSwikidesign
7f0fda08aSwikidesign// must be run within Dokuwiki
8f0fda08aSwikidesignif (!defined('DOKU_INC')) die();
9f0fda08aSwikidesign
10573e23a1Swikidesignif (!defined('DOKU_LF')) define('DOKU_LF', "\n");
11573e23a1Swikidesignif (!defined('DOKU_TAB')) define('DOKU_TAB', "\t");
12f0fda08aSwikidesignif (!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/');
13f0fda08aSwikidesign
14573e23a1Swikidesignrequire_once(DOKU_PLUGIN.'action.php');
15f0fda08aSwikidesign
16f0fda08aSwikidesignclass action_plugin_discussion extends DokuWiki_Action_Plugin{
17f0fda08aSwikidesign
18f1c6610eSpierre.spring    var $avatar = null;
19f1c6610eSpierre.spring    var $style = null;
20f1c6610eSpierre.spring    var $use_avatar = null;
21f1c6610eSpierre.spring
22f0fda08aSwikidesign    function getInfo() {
23f0fda08aSwikidesign        return array(
244c4d91c6SGina Haeussge                'author' => 'Gina Häußge, Michael Klier, Esther Brunner',
254c4d91c6SGina Haeussge                'email'  => 'dokuwiki@chimeric.de',
263648fb3eSMichael Klier                'date'   => @file_get_contents(DOKU_PLUGIN.'discussion/VERSION'),
27573e23a1Swikidesign                'name'   => 'Discussion Plugin (action component)',
28f0fda08aSwikidesign                'desc'   => 'Enables discussion features',
294c4d91c6SGina Haeussge                'url'    => 'http://wiki.splitbrain.org/plugin:discussion',
30f0fda08aSwikidesign                );
31f0fda08aSwikidesign    }
32f0fda08aSwikidesign
33f0fda08aSwikidesign    function register(&$contr) {
34f0fda08aSwikidesign        $contr->register_hook(
35f0fda08aSwikidesign                'ACTION_ACT_PREPROCESS',
36f0fda08aSwikidesign                'BEFORE',
37f0fda08aSwikidesign                $this,
38f0fda08aSwikidesign                'handle_act_preprocess',
39f0fda08aSwikidesign                array()
40f0fda08aSwikidesign                );
41f0fda08aSwikidesign        $contr->register_hook(
42f0fda08aSwikidesign                'TPL_ACT_RENDER',
43f0fda08aSwikidesign                'AFTER',
44f0fda08aSwikidesign                $this,
45f0fda08aSwikidesign                'comments',
46f0fda08aSwikidesign                array()
47f0fda08aSwikidesign                );
48479dd10fSwikidesign        $contr->register_hook(
49a1ca9e44Swikidesign                'INDEXER_PAGE_ADD',
50a1ca9e44Swikidesign                'AFTER',
51a1ca9e44Swikidesign                $this,
5261437513Swikidesign                'idx_add_discussion',
53a1ca9e44Swikidesign                array()
54a1ca9e44Swikidesign                );
552d4bee9aSMichael Klier        $contr->register_hook(
562d4bee9aSMichael Klier                'TPL_METAHEADER_OUTPUT',
572d4bee9aSMichael Klier                'BEFORE',
582d4bee9aSMichael Klier                $this,
592d4bee9aSMichael Klier                'handle_tpl_metaheader_output',
602d4bee9aSMichael Klier                array()
612d4bee9aSMichael Klier                );
622d4bee9aSMichael Klier        $contr->register_hook(
632d4bee9aSMichael Klier                'TOOLBAR_DEFINE',
642d4bee9aSMichael Klier                'AFTER',
652d4bee9aSMichael Klier                $this,
662d4bee9aSMichael Klier                'handle_toolbar_define',
672d4bee9aSMichael Klier                array()
682d4bee9aSMichael Klier                );
692d4bee9aSMichael Klier    }
702d4bee9aSMichael Klier
712d4bee9aSMichael Klier    /**
722d4bee9aSMichael Klier     * Modify Tollbar for use with discussion plugin
732d4bee9aSMichael Klier     *
742d4bee9aSMichael Klier     * @author Michael Klier <chi@chimeric.de>
752d4bee9aSMichael Klier     */
762d4bee9aSMichael Klier    function handle_toolbar_define(&$event, $param) {
772d4bee9aSMichael Klier        global $ACT;
782d4bee9aSMichael Klier        if($ACT != 'show') return;
792d4bee9aSMichael Klier
801de52da1SMichael Klier        if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) {
812d4bee9aSMichael Klier            $toolbar = array();
822d4bee9aSMichael Klier            foreach($event->data as $btn) {
832d4bee9aSMichael Klier                if($btn['type'] == 'mediapopup') continue;
842d4bee9aSMichael Klier                if($btn['type'] == 'signature') continue;
852d4bee9aSMichael Klier                if(preg_match("/=+?/", $btn['open'])) continue;
862d4bee9aSMichael Klier                array_push($toolbar, $btn);
872d4bee9aSMichael Klier            }
882d4bee9aSMichael Klier            $event->data = $toolbar;
892d4bee9aSMichael Klier        }
902d4bee9aSMichael Klier    }
912d4bee9aSMichael Klier
922d4bee9aSMichael Klier    /**
932d4bee9aSMichael Klier     * Dirty workaround to add a toolbar to the discussion plugin
942d4bee9aSMichael Klier     *
952d4bee9aSMichael Klier     * @author Michael Klier <chi@chimeric.de>
962d4bee9aSMichael Klier     */
972d4bee9aSMichael Klier    function handle_tpl_metaheader_output(&$event, $param) {
982d4bee9aSMichael Klier        global $ACT;
992d4bee9aSMichael Klier        global $ID;
1002d4bee9aSMichael Klier        if($ACT != 'show') return;
1012d4bee9aSMichael Klier
1022d4bee9aSMichael Klier        // FIXME check if this works for global discussion/on too
1031de52da1SMichael Klier        if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) {
1042d4bee9aSMichael Klier            // FIXME ugly workaround, replace this once DW the toolbar code is more flexible
1052d4bee9aSMichael Klier            array_unshift($event->data['script'], array('type' => 'text/javascript', 'charset' => 'utf-8', '_data' => '', 'src' => DOKU_BASE.'lib/scripts/edit.js'));
1062d4bee9aSMichael Klier            @require_once(DOKU_INC.'inc/toolbar.php');
1072d4bee9aSMichael Klier            ob_start();
1082d4bee9aSMichael Klier            print 'NS = "' . getNS($ID) . '";'; // we have to define NS, otherwise we get get JS errors
1092d4bee9aSMichael Klier            toolbar_JSdefines('toolbar');
1102d4bee9aSMichael Klier            $script = ob_get_clean();
1112d4bee9aSMichael Klier            array_push($event->data['script'], array('type' => 'text/javascript', 'charset' => "utf-8", '_data' => $script));
1122d4bee9aSMichael Klier        }
113f0fda08aSwikidesign    }
114f0fda08aSwikidesign
115f0fda08aSwikidesign    /**
116a1d93126SGina Haeussge     * Handles comment actions, dispatches data processing routines
117f0fda08aSwikidesign     */
118a1d93126SGina Haeussge    function handle_act_preprocess(&$event, $param) {
119284f9aa2SMichael Klier        global $ID;
120d7eb2e24SMichael Klier        global $INFO;
121284f9aa2SMichael Klier        global $conf;
1223011fb8bSMichael Klier        global $lang;
123573e23a1Swikidesign
124a1d93126SGina Haeussge        // handle newthread ACTs
125a1d93126SGina Haeussge        if ($event->data == 'newthread') {
126a1d93126SGina Haeussge            // we can handle it -> prevent others
127a1d93126SGina Haeussge            $event->preventDefault();
128a1d93126SGina Haeussge            $event->data = $this->_newThread();
129a1d93126SGina Haeussge        }
130a1d93126SGina Haeussge
131a1d93126SGina Haeussge        // enable captchas
132a1d93126SGina Haeussge        if ((in_array($_REQUEST['comment'], array('add', 'save')))
133a1d93126SGina Haeussge                && (@file_exists(DOKU_PLUGIN.'captcha/action.php'))) {
134a1d93126SGina Haeussge            $this->_captchaCheck();
135a1d93126SGina Haeussge        }
136a1d93126SGina Haeussge
1373011fb8bSMichael Klier        // if we are not in show mode or someone wants to unsubscribe, that was all for now
138*9881d835SMichael Klier        if ($event->data != 'show' && $event->data != 'unsubscribe' && $event->data != 'confirmsubscribe') return;
139a1d93126SGina Haeussge
140*9881d835SMichael Klier        if ($event->data == 'unsubscribe' or $event->data == 'confirmsubscribe') {
141*9881d835SMichael Klier            // ok we can handle it prevent others
142*9881d835SMichael Klier            $event->preventDefault();
143*9881d835SMichael Klier
1443011fb8bSMichael Klier            if (!isset($_REQUEST['hash'])) {
145*9881d835SMichael Klier                return false;
1463011fb8bSMichael Klier            } else {
1473011fb8bSMichael Klier                $file = metaFN($ID, '.comments');
1483011fb8bSMichael Klier                $data = unserialize(io_readFile($file));
149*9881d835SMichael Klier                foreach($data['subscribers'] as $mail => $info)  {
150*9881d835SMichael Klier                    // convert old style subscribers just in case
151*9881d835SMichael Klier                    if(!is_array($info)) {
152*9881d835SMichael Klier                        $hash = $data['subscribers'][$mail];
153*9881d835SMichael Klier                        $data['subscribers'][$mail]['hash']   = $hash;
154*9881d835SMichael Klier                        $data['subscribers'][$mail]['active'] = true;
155*9881d835SMichael Klier                        $data['subscribers'][$mail]['confirmsent'] = true;
156*9881d835SMichael Klier                    }
157*9881d835SMichael Klier                }
158*9881d835SMichael Klier
159*9881d835SMichael Klier                if($data['subscribers'][$mail]['hash'] == $_REQUEST['hash']) {
160*9881d835SMichael Klier                    if($event->data == 'unsubscribe') {
1613011fb8bSMichael Klier                        unset($data['subscribers'][$mail]);
1623011fb8bSMichael Klier                        msg(sprintf($lang['unsubscribe_success'], $mail, $ID), 1);
163*9881d835SMichael Klier                    } elseif($event->data == 'confirmsubscribe') {
164*9881d835SMichael Klier                        $data['subscribers'][$mail]['active'] = true;
165*9881d835SMichael Klier                        msg(sprintf($lang['subscribe_success'], $mail, $ID), 1);
166*9881d835SMichael Klier                    }
167*9881d835SMichael Klier                    io_saveFile($file, serialize($data));
1683011fb8bSMichael Klier                    $event->data = 'show';
1693011fb8bSMichael Klier                    return true;
170*9881d835SMichael Klier                } else {
1713011fb8bSMichael Klier                    return false;
1723011fb8bSMichael Klier                }
173*9881d835SMichael Klier            }
1743011fb8bSMichael Klier        } else {
175a1d93126SGina Haeussge            // do the data processing for comments
176f0fda08aSwikidesign            $cid  = $_REQUEST['cid'];
177f0fda08aSwikidesign            switch ($_REQUEST['comment']) {
178f0fda08aSwikidesign                case 'add':
17994c5d164SMichael Klier                    if(isset($_SERVER['REMOTE_USER']) && !$this->getConf('adminimport')) {
18094c5d164SMichael Klier                        $comment['user']['id'] = $_SERVER['REMOTE_USER'];
18194c5d164SMichael Klier                        $comment['user']['name'] = $INFO['userinfo']['name'];
18294c5d164SMichael Klier                        $comment['user']['mail'] = $INFO['userinfo']['mail'];
18394c5d164SMichael Klier                    } elseif((isset($_SERVER['REMOTE_USER']) && $this->getConf('adminimport') && auth_ismanager()) || !isset($_SERVER['REMOTE_USER'])) {
18494c5d164SMichael Klier                        $comment['user']['id'] = 'test'.hsc($_REQUEST['user']);
18594c5d164SMichael Klier                        $comment['user']['name'] = hsc($_REQUEST['name']);
18694c5d164SMichael Klier                        $comment['user']['mail'] = hsc($_REQUEST['mail']);
18794c5d164SMichael Klier                    }
18894c5d164SMichael Klier                    $comment['user']['address'] = ($this->getConf('addressfield')) ? hsc($_REQUEST['address']) : '';
189a10b5c98SMichael Klier                    $comment['user']['url'] = ($this->getConf('urlfield')) ? $this->_checkURL($_REQUEST['url']) : '';
19094c5d164SMichael Klier                    $comment['subscribe'] = ($this->getConf('subscribe')) ? $_REQUEST['subscribe'] : '';
19194c5d164SMichael Klier                    $comment['date'] = array('created' => $_REQUEST['date']);
19294c5d164SMichael Klier                    $comment['raw'] = cleanText($_REQUEST['text']);
193f0fda08aSwikidesign                    $repl = $_REQUEST['reply'];
194f0fda08aSwikidesign                    $this->_add($comment, $repl);
195f0fda08aSwikidesign                    break;
196f0fda08aSwikidesign
197f0fda08aSwikidesign                case 'save':
198f0fda08aSwikidesign                    $raw  = cleanText($_REQUEST['text']);
199264b7327Swikidesign                    $this->_save(array($cid), $raw);
200f0fda08aSwikidesign                    break;
201f0fda08aSwikidesign
2021e46d176Swikidesign                case 'delete':
203264b7327Swikidesign                    $this->_save(array($cid), '');
2042ee3dca3Swikidesign                    break;
2051e46d176Swikidesign
206f0fda08aSwikidesign                case 'toogle':
207264b7327Swikidesign                    $this->_save(array($cid), '', 'toogle');
208f0fda08aSwikidesign                    break;
209a1d93126SGina Haeussge            }
2103011fb8bSMichael Klier        }
211284f9aa2SMichael Klier
212284f9aa2SMichael Klier        // FIXME use new TPL_TOC_RENDER event in the future
213d7eb2e24SMichael Klier        if(count($INFO['meta']['description']['tableofcontents']) >= ($conf['maxtoclevel']-1) && $INFO['meta']['internal']['toc']) {
214284f9aa2SMichael Klier
215284f9aa2SMichael Klier            $TOC = array();
216284f9aa2SMichael Klier            global $TOC;
217d7eb2e24SMichael Klier            $TOC = $INFO['meta']['description']['tableofcontents'];
218284f9aa2SMichael Klier
219284f9aa2SMichael Klier            $tocitem = array( 'hid' => 'discussion__section',
220284f9aa2SMichael Klier                              'title' => $this->getLang('discussion'),
221284f9aa2SMichael Klier                              'type' => 'ul',
222284f9aa2SMichael Klier                              'level' => 1 );
223284f9aa2SMichael Klier
224284f9aa2SMichael Klier            $file = metaFN($ID, '.comments');
225284f9aa2SMichael Klier            if(@file_exists($file)) {
226284f9aa2SMichael Klier                $data = unserialize(io_readFile($file));
227284f9aa2SMichael Klier                if($data['status'] != 0 && !empty($TOC)) {
228284f9aa2SMichael Klier                    $TOC[] = $tocitem;
229284f9aa2SMichael Klier                }
230284f9aa2SMichael Klier            }
231284f9aa2SMichael Klier        }
232a1d93126SGina Haeussge    }
233a1d93126SGina Haeussge
234a1d93126SGina Haeussge    /**
235a1d93126SGina Haeussge     * Main function; dispatches the visual comment actions
236a1d93126SGina Haeussge     */
237a1d93126SGina Haeussge    function comments(&$event, $param) {
238a1d93126SGina Haeussge        if ($event->data != 'show') return; // nothing to do for us
239a1d93126SGina Haeussge
240a1d93126SGina Haeussge        $cid  = $_REQUEST['cid'];
241a1d93126SGina Haeussge        switch ($_REQUEST['comment']) {
242a1d93126SGina Haeussge            case 'edit':
243a1d93126SGina Haeussge                $this->_show(NULL, $cid);
244a1d93126SGina Haeussge                break;
2452b18adb9SMichael Klier            default:
246f0fda08aSwikidesign                $this->_show($cid);
2472b18adb9SMichael Klier                break;
248f0fda08aSwikidesign        }
249f0fda08aSwikidesign    }
250f0fda08aSwikidesign
251f0fda08aSwikidesign    /**
252a1d93126SGina Haeussge     * Redirects browser to given comment anchor
253a1d93126SGina Haeussge     */
254a1d93126SGina Haeussge    function _redirect($cid) {
255a1d93126SGina Haeussge        global $ID;
256a1d93126SGina Haeussge        global $ACT;
257a1d93126SGina Haeussge
258a1d93126SGina Haeussge        if ($ACT !== 'show') return;
2591810ba9cSMichael Klier        header('Location: ' . wl($ID) . '#comment_' . $cid);
2606d1f4f20SGina Haeussge        exit();
261a1d93126SGina Haeussge    }
262a1d93126SGina Haeussge
263a1d93126SGina Haeussge    /**
264f0fda08aSwikidesign     * Shows all comments of the current page
265f0fda08aSwikidesign     */
266f0fda08aSwikidesign    function _show($reply = NULL, $edit = NULL) {
2672b18adb9SMichael Klier        global $ID;
2682b18adb9SMichael Klier        global $INFO;
2692b18adb9SMichael Klier        global $ACT;
270573e23a1Swikidesign
271479dd10fSwikidesign        // get .comments meta file name
272f0fda08aSwikidesign        $file = metaFN($ID, '.comments');
273f0fda08aSwikidesign
2742b18adb9SMichael Klier        if (!@file_exists($file) && !$this->getConf('automatic')) return false;
275f0fda08aSwikidesign
2762b18adb9SMichael Klier        // load data
2772b18adb9SMichael Klier        if (@file_exists($file)) {
2782b18adb9SMichael Klier            $data = unserialize(io_readFile($file, false));
279479dd10fSwikidesign            if (!$data['status']) return false; // comments are turned off
2802b18adb9SMichael Klier        } elseif (!@file_exists($file) && $this->getConf('automatic') && $INFO['exists']) {
2812b18adb9SMichael Klier            // set status to show the comment form
2822b18adb9SMichael Klier            $data['status'] = 1;
2832b18adb9SMichael Klier            $data['number'] = 0;
2842b18adb9SMichael Klier        }
285f0fda08aSwikidesign
286f0fda08aSwikidesign        // section title
28707c376bbSwikidesign        $title = ($data['title'] ? hsc($data['title']) : $this->getLang('discussion'));
2884a0a1bd2Swikidesign        ptln('<div class="comment_wrapper">');
2894a0a1bd2Swikidesign        ptln('<h2><a name="discussion__section" id="discussion__section">', 2);
2904a0a1bd2Swikidesign        ptln($title, 4);
2914a0a1bd2Swikidesign        ptln('</a></h2>', 2);
2924a0a1bd2Swikidesign        ptln('<div class="level2 hfeed">', 2);
293f0fda08aSwikidesign        // now display the comments
294f0fda08aSwikidesign        if (isset($data['comments'])) {
29531aab30eSGina Haeussge            if (!$this->getConf('usethreading')) {
29631aab30eSGina Haeussge                $data['comments'] = $this->_flattenThreads($data['comments']);
29731aab30eSGina Haeussge                uasort($data['comments'], '_sortCallBack');
29831aab30eSGina Haeussge            }
299dbd9d5cdSMichael Klier            if($this->getConf('newestfirst')) {
300dbd9d5cdSMichael Klier                $data['comments'] = array_reverse($data['comments']);
301dbd9d5cdSMichael Klier            }
302f0fda08aSwikidesign            foreach ($data['comments'] as $key => $value) {
303f0fda08aSwikidesign                if ($key == $edit) $this->_form($value['raw'], 'save', $edit); // edit form
304f0fda08aSwikidesign                else $this->_print($key, $data, '', $reply);
305f0fda08aSwikidesign            }
306f0fda08aSwikidesign        }
307f0fda08aSwikidesign
308f0fda08aSwikidesign        // comment form
30931aab30eSGina Haeussge        if (($data['status'] == 1) && (!$reply || !$this->getConf('usethreading')) && !$edit) $this->_form('');
310f0fda08aSwikidesign
3114a0a1bd2Swikidesign        ptln('</div>', 2); // level2 hfeed
3124a0a1bd2Swikidesign        ptln('</div>'); // comment_wrapper
313f0fda08aSwikidesign
314f0fda08aSwikidesign        return true;
315f0fda08aSwikidesign    }
316f0fda08aSwikidesign
31731aab30eSGina Haeussge    function _flattenThreads($comments, $keys = null) {
31831aab30eSGina Haeussge        if (is_null($keys))
31931aab30eSGina Haeussge            $keys = array_keys($comments);
32031aab30eSGina Haeussge
32131aab30eSGina Haeussge        foreach($keys as $cid) {
32231aab30eSGina Haeussge            if (!empty($comments[$cid]['replies'])) {
32331aab30eSGina Haeussge                $rids = $comments[$cid]['replies'];
32431aab30eSGina Haeussge                $comments = $this->_flattenThreads($comments, $rids);
32531aab30eSGina Haeussge                $comments[$cid]['replies'] = array();
32631aab30eSGina Haeussge            }
32731aab30eSGina Haeussge            $comments[$cid]['parent'] = '';
32831aab30eSGina Haeussge        }
32931aab30eSGina Haeussge        return $comments;
33031aab30eSGina Haeussge    }
33131aab30eSGina Haeussge
332f0fda08aSwikidesign    /**
333f0fda08aSwikidesign     * Adds a new comment and then displays all comments
334f0fda08aSwikidesign     */
335f0fda08aSwikidesign    function _add($comment, $parent) {
3363011fb8bSMichael Klier        global $lang;
3373011fb8bSMichael Klier        global $ID;
3383011fb8bSMichael Klier        global $TEXT;
339f0fda08aSwikidesign
340f0fda08aSwikidesign        $otxt = $TEXT; // set $TEXT to comment text for wordblock check
341f0fda08aSwikidesign        $TEXT = $comment['raw'];
342f0fda08aSwikidesign
343f0fda08aSwikidesign        // spamcheck against the DokuWiki blacklist
344f0fda08aSwikidesign        if (checkwordblock()) {
345f0fda08aSwikidesign            msg($this->getLang('wordblock'), -1);
346f0fda08aSwikidesign            return false;
347f0fda08aSwikidesign        }
348f0fda08aSwikidesign
3493011fb8bSMichael Klier        if ((!$this->getConf('allowguests'))
3503011fb8bSMichael Klier                && ($comment['user']['id'] != $_SERVER['REMOTE_USER']))
3513011fb8bSMichael Klier            return false; // guest comments not allowed
3523011fb8bSMichael Klier
353f0fda08aSwikidesign        $TEXT = $otxt; // restore global $TEXT
354f0fda08aSwikidesign
355f0fda08aSwikidesign        // get discussion meta file name
356f0fda08aSwikidesign        $file = metaFN($ID, '.comments');
357f0fda08aSwikidesign
3582b18adb9SMichael Klier        // create comments file if it doesn't exist yet
3592b18adb9SMichael Klier        if(!@file_exists($file)) {
3602b18adb9SMichael Klier            $data = array('status' => 1, 'number' => 0);
3612b18adb9SMichael Klier            io_saveFile($file, serialize($data));
3622b18adb9SMichael Klier        } else {
363f0fda08aSwikidesign            $data = array();
364f0fda08aSwikidesign            $data = unserialize(io_readFile($file, false));
365f0fda08aSwikidesign            if ($data['status'] != 1) return false; // comments off or closed
3662b18adb9SMichael Klier        }
3672b18adb9SMichael Klier
3683011fb8bSMichael Klier        if ($comment['date']['created']) {
3693011fb8bSMichael Klier            $date = strtotime($comment['date']['created']);
3703011fb8bSMichael Klier        } else {
3713011fb8bSMichael Klier            $date = time();
3723011fb8bSMichael Klier        }
373f0fda08aSwikidesign
3743011fb8bSMichael Klier        if ($date == -1) {
3753011fb8bSMichael Klier            $date = time();
3763011fb8bSMichael Klier        }
3773011fb8bSMichael Klier
3786046f25cSwikidesign        $cid  = md5($comment['user']['id'].$date); // create a unique id
379f0fda08aSwikidesign
3803011fb8bSMichael Klier        if (!is_array($data['comments'][$parent])) {
3813011fb8bSMichael Klier            $parent = NULL; // invalid parent comment
3823011fb8bSMichael Klier        }
383f0fda08aSwikidesign
384f0fda08aSwikidesign        // render the comment
385f0fda08aSwikidesign        $xhtml = $this->_render($comment['raw']);
386f0fda08aSwikidesign
387f0fda08aSwikidesign        // fill in the new comment
388f0fda08aSwikidesign        $data['comments'][$cid] = array(
3896046f25cSwikidesign                'user'    => $comment['user'],
3906046f25cSwikidesign                'date'    => array('created' => $date),
391f0fda08aSwikidesign                'show'    => true,
3926046f25cSwikidesign                'raw'     => $comment['raw'],
393f0fda08aSwikidesign                'xhtml'   => $xhtml,
394f0fda08aSwikidesign                'parent'  => $parent,
395f0fda08aSwikidesign                'replies' => array()
396f0fda08aSwikidesign                );
397f0fda08aSwikidesign
3983011fb8bSMichael Klier        if($comment['subscribe']) {
3993011fb8bSMichael Klier            $mail = $comment['user']['mail'];
4003011fb8bSMichael Klier            if($data['subscribers']) {
4013011fb8bSMichael Klier                if(!$data['subscribers'][$mail]) {
402*9881d835SMichael Klier                    $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand());
403*9881d835SMichael Klier                    $data['subscribers'][$mail]['active'] = false;
404*9881d835SMichael Klier                    $data['subscribers'][$mail]['confirmsent'] = false;
405*9881d835SMichael Klier                } else {
406*9881d835SMichael Klier                    // convert old style subscribers and set them active
407*9881d835SMichael Klier                    if(!is_array($data['subscribers'][$mail])) {
408*9881d835SMichael Klier                        $hash = $data['subscribers'][$mail];
409*9881d835SMichael Klier                        $data['subscribers'][$mail]['hash'] = $hash;
410*9881d835SMichael Klier                        $data['subscribers'][$mail]['active'] = true;
411*9881d835SMichael Klier                        $data['subscribers'][$mail]['confirmsent'] = true;
412*9881d835SMichael Klier                    }
4133011fb8bSMichael Klier                }
4143011fb8bSMichael Klier            } else {
415*9881d835SMichael Klier                $data['subscribers'][$mail]['hash']   = md5($mail . mt_rand());
416*9881d835SMichael Klier                $data['subscribers'][$mail]['active'] = false;
417*9881d835SMichael Klier                $data['subscribers'][$mail]['confirmsent'] = false;
4183011fb8bSMichael Klier            }
4193011fb8bSMichael Klier        }
4203011fb8bSMichael Klier
421f0fda08aSwikidesign        // update parent comment
422f0fda08aSwikidesign        if ($parent) $data['comments'][$parent]['replies'][] = $cid;
423f0fda08aSwikidesign
424f0fda08aSwikidesign        // update the number of comments
425f0fda08aSwikidesign        $data['number']++;
426f0fda08aSwikidesign
427f0fda08aSwikidesign        // notify subscribers of the page
4288b42cefbSMichael Klier        $data['comments'][$cid]['cid'] = $cid;
4293011fb8bSMichael Klier        $this->_notify($data['comments'][$cid], $data['subscribers']);
430f0fda08aSwikidesign
431*9881d835SMichael Klier        // save the comment metadata file
432*9881d835SMichael Klier        io_saveFile($file, serialize($data));
433*9881d835SMichael Klier        $this->_addLogEntry($date, $ID, 'cc', '', $cid);
434*9881d835SMichael Klier
435a1d93126SGina Haeussge        $this->_redirect($cid);
436f0fda08aSwikidesign        return true;
437f0fda08aSwikidesign    }
438f0fda08aSwikidesign
439f0fda08aSwikidesign    /**
440f0fda08aSwikidesign     * Saves the comment with the given ID and then displays all comments
441f0fda08aSwikidesign     */
442264b7327Swikidesign    function _save($cids, $raw, $act = NULL) {
443a1ace3c9Swikidesign        global $ID;
444f0fda08aSwikidesign
4452ee3dca3Swikidesign        if ($raw) {
4462ee3dca3Swikidesign            global $TEXT;
4472ee3dca3Swikidesign
448f0fda08aSwikidesign            $otxt = $TEXT; // set $TEXT to comment text for wordblock check
449f0fda08aSwikidesign            $TEXT = $raw;
450f0fda08aSwikidesign
451f0fda08aSwikidesign            // spamcheck against the DokuWiki blacklist
452f0fda08aSwikidesign            if (checkwordblock()) {
453f0fda08aSwikidesign                msg($this->getLang('wordblock'), -1);
454f0fda08aSwikidesign                return false;
455f0fda08aSwikidesign            }
456f0fda08aSwikidesign
457f0fda08aSwikidesign            $TEXT = $otxt; // restore global $TEXT
4582ee3dca3Swikidesign        }
459f0fda08aSwikidesign
460f0fda08aSwikidesign        // get discussion meta file name
461f0fda08aSwikidesign        $file = metaFN($ID, '.comments');
462f0fda08aSwikidesign        $data = unserialize(io_readFile($file, false));
463f0fda08aSwikidesign
464573e23a1Swikidesign        if (!is_array($cids)) $cids = array($cids);
465264b7327Swikidesign        foreach ($cids as $cid) {
466264b7327Swikidesign
4676046f25cSwikidesign            if (is_array($data['comments'][$cid]['user'])) {
4686046f25cSwikidesign                $user    = $data['comments'][$cid]['user']['id'];
4696046f25cSwikidesign                $convert = false;
4706046f25cSwikidesign            } else {
4716046f25cSwikidesign                $user    = $data['comments'][$cid]['user'];
4726046f25cSwikidesign                $convert = true;
4736046f25cSwikidesign            }
4746046f25cSwikidesign
475f0fda08aSwikidesign            // someone else was trying to edit our comment -> abort
4763e02b3ffSwikidesign            if (($user != $_SERVER['REMOTE_USER']) && (!auth_ismanager())) return false;
477f0fda08aSwikidesign
478f0fda08aSwikidesign            $date = time();
479f0fda08aSwikidesign
4806046f25cSwikidesign            // need to convert to new format?
4816046f25cSwikidesign            if ($convert) {
4826046f25cSwikidesign                $data['comments'][$cid]['user'] = array(
4836046f25cSwikidesign                        'id'      => $user,
4846046f25cSwikidesign                        'name'    => $data['comments'][$cid]['name'],
4856046f25cSwikidesign                        'mail'    => $data['comments'][$cid]['mail'],
4866046f25cSwikidesign                        'url'     => $data['comments'][$cid]['url'],
4876046f25cSwikidesign                        'address' => $data['comments'][$cid]['address'],
4886046f25cSwikidesign                        );
4896046f25cSwikidesign                $data['comments'][$cid]['date'] = array(
4906046f25cSwikidesign                        'created' => $data['comments'][$cid]['date']
4916046f25cSwikidesign                        );
4926046f25cSwikidesign            }
4936046f25cSwikidesign
494264b7327Swikidesign            if ($act == 'toogle') {     // toogle visibility
495f0fda08aSwikidesign                $now = $data['comments'][$cid]['show'];
496f0fda08aSwikidesign                $data['comments'][$cid]['show'] = !$now;
497f0fda08aSwikidesign                $data['number'] = $this->_count($data);
498f0fda08aSwikidesign
499f0fda08aSwikidesign                $type = ($data['comments'][$cid]['show'] ? 'sc' : 'hc');
500f0fda08aSwikidesign
501264b7327Swikidesign            } elseif ($act == 'show') { // show comment
502264b7327Swikidesign                $data['comments'][$cid]['show'] = true;
503264b7327Swikidesign                $data['number'] = $this->_count($data);
504264b7327Swikidesign
505573e23a1Swikidesign                $type = 'sc'; // show comment
506264b7327Swikidesign
507264b7327Swikidesign            } elseif ($act == 'hide') { // hide comment
508264b7327Swikidesign                $data['comments'][$cid]['show'] = false;
509264b7327Swikidesign                $data['number'] = $this->_count($data);
510264b7327Swikidesign
511573e23a1Swikidesign                $type = 'hc'; // hide comment
512264b7327Swikidesign
513f0fda08aSwikidesign            } elseif (!$raw) {          // remove the comment
514efbe59d0Swikidesign                $data['comments'] = $this->_removeComment($cid, $data['comments']);
515f0fda08aSwikidesign                $data['number'] = $this->_count($data);
516f0fda08aSwikidesign
517573e23a1Swikidesign                $type = 'dc'; // delete comment
518f0fda08aSwikidesign
519f0fda08aSwikidesign            } else {                   // save changed comment
520f0fda08aSwikidesign                $xhtml = $this->_render($raw);
521f0fda08aSwikidesign
522f0fda08aSwikidesign                // now change the comment's content
5236046f25cSwikidesign                $data['comments'][$cid]['date']['modified'] = $date;
5246046f25cSwikidesign                $data['comments'][$cid]['raw']              = $raw;
525f0fda08aSwikidesign                $data['comments'][$cid]['xhtml']            = $xhtml;
526f0fda08aSwikidesign
527573e23a1Swikidesign                $type = 'ec'; // edit comment
528f0fda08aSwikidesign            }
529264b7327Swikidesign        }
530264b7327Swikidesign
531f0fda08aSwikidesign        // save the comment metadata file
532f0fda08aSwikidesign        io_saveFile($file, serialize($data));
533f0fda08aSwikidesign        $this->_addLogEntry($date, $ID, $type, '', $cid);
534f0fda08aSwikidesign
535a1d93126SGina Haeussge        $this->_redirect($cid);
536f0fda08aSwikidesign        return true;
537f0fda08aSwikidesign    }
538f0fda08aSwikidesign
539f0fda08aSwikidesign    /**
540efbe59d0Swikidesign     * Recursive function to remove a comment
541efbe59d0Swikidesign     */
542efbe59d0Swikidesign    function _removeComment($cid, $comments) {
543efbe59d0Swikidesign        if (is_array($comments[$cid]['replies'])) {
544efbe59d0Swikidesign            foreach ($comments[$cid]['replies'] as $rid) {
545efbe59d0Swikidesign                $comments = $this->_removeComment($rid, $comments);
546efbe59d0Swikidesign            }
547efbe59d0Swikidesign        }
548efbe59d0Swikidesign        unset($comments[$cid]);
549efbe59d0Swikidesign        return $comments;
550efbe59d0Swikidesign    }
551efbe59d0Swikidesign
552efbe59d0Swikidesign    /**
553f0fda08aSwikidesign     * Prints an individual comment
554f0fda08aSwikidesign     */
555f0fda08aSwikidesign    function _print($cid, &$data, $parent = '', $reply = '', $visible = true) {
556f0fda08aSwikidesign
5572ee3dca3Swikidesign        if (!isset($data['comments'][$cid])) return false; // comment was removed
558f0fda08aSwikidesign        $comment = $data['comments'][$cid];
559f0fda08aSwikidesign
560f0fda08aSwikidesign        if (!is_array($comment)) return false;             // corrupt datatype
561f0fda08aSwikidesign
562f0fda08aSwikidesign        if ($comment['parent'] != $parent) return true;    // reply to an other comment
563f0fda08aSwikidesign
564f0fda08aSwikidesign        if (!$comment['show']) {                            // comment hidden
565a1ace3c9Swikidesign            if (auth_ismanager()) $hidden = ' comment_hidden';
566f0fda08aSwikidesign            else return true;
5674a0a1bd2Swikidesign        } else {
5684a0a1bd2Swikidesign            $hidden = '';
569f0fda08aSwikidesign        }
570f0fda08aSwikidesign
571f1c6610eSpierre.spring        if($this->getConf('newestfirst')) {
572f1c6610eSpierre.spring            // reply form
573f1c6610eSpierre.spring            $this->_print_form($cid, $reply);
574f1c6610eSpierre.spring            // replies to this comment entry?
575f1c6610eSpierre.spring            $this->_print_replies($cid, $data, $reply, $visible);
576f1c6610eSpierre.spring            // print the actual comment
577f1c6610eSpierre.spring            $this->_print_comment($cid, &$data, $parent, $reply, $visible, $hidden);
578f1c6610eSpierre.spring        } else {
579f1c6610eSpierre.spring            // print the actual comment
580f1c6610eSpierre.spring            $this->_print_comment($cid, &$data, $parent, $reply, $visible, $hidden);
581f1c6610eSpierre.spring            // replies to this comment entry?
582f1c6610eSpierre.spring            $this->_print_replies($cid, $data, $reply, $visible);
583f1c6610eSpierre.spring            // reply form
584f1c6610eSpierre.spring            $this->_print_form($cid, $reply);
585f1c6610eSpierre.spring        }
586f1c6610eSpierre.spring    }
587f1c6610eSpierre.spring
588f1c6610eSpierre.spring    function _print_comment($cid, &$data, $parent, $reply, $visible, $hidden)
589f1c6610eSpierre.spring    {
590f1c6610eSpierre.spring        global $conf, $lang, $ID, $HIGH;
591f1c6610eSpierre.spring        $comment = $data['comments'][$cid];
592f1c6610eSpierre.spring
593f0fda08aSwikidesign        // comment head with date and user data
5944a0a1bd2Swikidesign        ptln('<div class="hentry'.$hidden.'">', 4);
5954a0a1bd2Swikidesign        ptln('<div class="comment_head">', 6);
5961810ba9cSMichael Klier        ptln('<a name="comment_'.$cid.'" id="comment_'.$cid.'"></a>', 8);
5974a0a1bd2Swikidesign        $head = '<span class="vcard author">';
598f0fda08aSwikidesign
5996046f25cSwikidesign        // prepare variables
6006046f25cSwikidesign        if (is_array($comment['user'])) { // new format
6016046f25cSwikidesign            $user    = $comment['user']['id'];
6026046f25cSwikidesign            $name    = $comment['user']['name'];
6036046f25cSwikidesign            $mail    = $comment['user']['mail'];
6046046f25cSwikidesign            $url     = $comment['user']['url'];
6056046f25cSwikidesign            $address = $comment['user']['address'];
6066046f25cSwikidesign        } else {                         // old format
6076046f25cSwikidesign            $user    = $comment['user'];
6086046f25cSwikidesign            $name    = $comment['name'];
6096046f25cSwikidesign            $mail    = $comment['mail'];
6106046f25cSwikidesign            $url     = $comment['url'];
6116046f25cSwikidesign            $address = $comment['address'];
6126046f25cSwikidesign        }
6136046f25cSwikidesign        if (is_array($comment['date'])) { // new format
6146046f25cSwikidesign            $created  = $comment['date']['created'];
6156046f25cSwikidesign            $modified = $comment['date']['modified'];
6166046f25cSwikidesign        } else {                         // old format
6176046f25cSwikidesign            $created  = $comment['date'];
6186046f25cSwikidesign            $modified = $comment['edited'];
6196046f25cSwikidesign        }
6206046f25cSwikidesign
621ce7b17cfSwikidesign        // show avatar image?
622f1c6610eSpierre.spring        if ($this->_use_avatar()) {
62382c01980SMichael Klier
62482c01980SMichael Klier            $files = @glob(mediaFN('user') . '/' . $user . '.*');
62582c01980SMichael Klier            if ($files) {
62682c01980SMichael Klier                foreach ($files as $file) {
62782c01980SMichael Klier                    if (preg_match('/jpg|jpeg|gif|png/', $file)) {
628f1c6610eSpierre.spring                        $head .= $this->avatar->getXHTML($user, $name, 'left');
62982c01980SMichael Klier                        break;
63082c01980SMichael Klier                    }
63182c01980SMichael Klier                }
63282c01980SMichael Klier            } elseif ($mail) {
633f1c6610eSpierre.spring                $head .= $this->avatar->getXHTML($mail, $name, 'left');
63482c01980SMichael Klier            } else {
635f1c6610eSpierre.spring                $head .= $this->avatar->getXHTML($user, $name, 'left');
63682c01980SMichael Klier            }
637f0fda08aSwikidesign        }
638f0fda08aSwikidesign
6396046f25cSwikidesign        if ($this->getConf('linkemail') && $mail) {
6404a0a1bd2Swikidesign            $head .= $this->email($mail, $name, 'email fn');
6416046f25cSwikidesign        } elseif ($url) {
642ab7c9f30SMichael Klier            $head .= $this->external_link($this->_checkURL($url), $name, 'urlextern url fn');
643f0fda08aSwikidesign        } else {
6444a0a1bd2Swikidesign            $head .= '<span class="fn">'.$name.'</span>';
645f0fda08aSwikidesign        }
6464a0a1bd2Swikidesign        if ($address) $head .= ', <span class="adr">'.$address.'</span>';
6474a0a1bd2Swikidesign        $head .= '</span>, '.
648f014bc86SMichael Klier            '<abbr class="published" title="'.strftime('%Y-%m-%dT%H:%M:%SZ', $created).'">'.
649f014bc86SMichael Klier            strftime($conf['dformat'], $created).'</abbr>';
6504a0a1bd2Swikidesign        if ($comment['edited']) $head .= ' (<abbr class="updated" title="'.
651f014bc86SMichael Klier                strftime('%Y-%m-%dT%H:%M:%SZ', $modified).'">'.strftime($conf['dformat'], $modified).
6526046f25cSwikidesign                '</abbr>)';
653f014bc86SMichael Klier        ptln($head, 8);
6544a0a1bd2Swikidesign        ptln('</div>', 6); // class="comment_head"
655f0fda08aSwikidesign
656f0fda08aSwikidesign        // main comment content
6574a0a1bd2Swikidesign        ptln('<div class="comment_body entry-content"'.
658f1c6610eSpierre.spring                ($this->getConf('useavatar') ? $this->_get_style() : '').'>', 6);
65915cdad37Spierre.spring        echo ($HIGH?html_hilight($comment['xhtml'],$HIGH):$comment['xhtml']).DOKU_LF;
6604a0a1bd2Swikidesign        ptln('</div>', 6); // class="comment_body"
661f0fda08aSwikidesign
662f0fda08aSwikidesign        if ($visible) {
6631184c36aSwikidesign            ptln('<div class="comment_buttons">', 6);
664f0fda08aSwikidesign
665f0fda08aSwikidesign            // show reply button?
666f1c4aa1aSwikidesign            if (($data['status'] == 1) && !$reply && $comment['show']
66731aab30eSGina Haeussge                    && ($this->getConf('allowguests') || $_SERVER['REMOTE_USER']) && $this->getConf('usethreading'))
6681e46d176Swikidesign                $this->_button($cid, $this->getLang('btn_reply'), 'reply', true);
669f0fda08aSwikidesign
6701184c36aSwikidesign            // show edit, show/hide and delete button?
6711184c36aSwikidesign            if ((($user == $_SERVER['REMOTE_USER']) && ($user != '')) || (auth_ismanager())) {
6721e46d176Swikidesign                $this->_button($cid, $lang['btn_secedit'], 'edit', true);
6731184c36aSwikidesign                $label = ($comment['show'] ? $this->getLang('btn_hide') : $this->getLang('btn_show'));
6741184c36aSwikidesign                $this->_button($cid, $label, 'toogle');
6751184c36aSwikidesign                $this->_button($cid, $lang['btn_delete'], 'delete');
676f0fda08aSwikidesign            }
6771184c36aSwikidesign            ptln('</div>', 6); // class="comment_buttons"
6781184c36aSwikidesign        }
6791184c36aSwikidesign        ptln('</div>', 4); // class="hentry"
680f0fda08aSwikidesign    }
681f0fda08aSwikidesign
682f1c6610eSpierre.spring    function _print_form($cid, $reply)
683f1c6610eSpierre.spring    {
68431aab30eSGina Haeussge        if ($this->getConf('usethreading') && $reply == $cid) {
6854a0a1bd2Swikidesign            ptln('<div class="comment_replies">', 4);
686f0fda08aSwikidesign            $this->_form('', 'add', $cid);
6874a0a1bd2Swikidesign            ptln('</div>', 4); // class="comment_replies"
688f0fda08aSwikidesign        }
689f0fda08aSwikidesign    }
690f0fda08aSwikidesign
691f1c6610eSpierre.spring    function _print_replies($cid, &$data, $reply, &$visible)
692f1c6610eSpierre.spring    {
693f1c6610eSpierre.spring        $comment = $data['comments'][$cid];
694f1c6610eSpierre.spring        if (!count($comment['replies'])) {
695f1c6610eSpierre.spring            return;
696f1c6610eSpierre.spring        }
697f1c6610eSpierre.spring        ptln('<div class="comment_replies"'.$this->_get_style().'>', 4);
698f1c6610eSpierre.spring        $visible = ($comment['show'] && $visible);
699f1c6610eSpierre.spring        foreach ($comment['replies'] as $rid) {
700f1c6610eSpierre.spring            $this->_print($rid, $data, $cid, $reply, $visible);
701f1c6610eSpierre.spring        }
702f1c6610eSpierre.spring        ptln('</div>', 4);
703f1c6610eSpierre.spring    }
704f1c6610eSpierre.spring
705f1c6610eSpierre.spring    function _use_avatar()
706f1c6610eSpierre.spring    {
707f1c6610eSpierre.spring        if (is_null($this->use_avatar)) {
708f1c6610eSpierre.spring            $this->use_avatar = $this->getConf('useavatar')
709f1c6610eSpierre.spring                    && (!plugin_isdisabled('avatar'))
710f1c6610eSpierre.spring                    && ($this->avatar =& plugin_load('helper', 'avatar'));
711f1c6610eSpierre.spring        }
712f1c6610eSpierre.spring        return $this->use_avatar;
713f1c6610eSpierre.spring    }
714f1c6610eSpierre.spring
715f1c6610eSpierre.spring    function _get_style()
716f1c6610eSpierre.spring    {
717f1c6610eSpierre.spring        if (is_null($this->style)){
718f1c6610eSpierre.spring            if ($this->_use_avatar()) {
719f1c6610eSpierre.spring                $this->style = ' style="margin-left: '.($this->avatar->getConf('size') + 14).'px;"';
720f1c6610eSpierre.spring            } else {
721f1c6610eSpierre.spring                $this->style = ' style="margin-left: 20px;"';
722f1c6610eSpierre.spring            }
723f1c6610eSpierre.spring        }
724f1c6610eSpierre.spring        return $this->style;
725f1c6610eSpierre.spring    }
726f1c6610eSpierre.spring
727f0fda08aSwikidesign    /**
728f0fda08aSwikidesign     * Outputs the comment form
729f0fda08aSwikidesign     */
730f0fda08aSwikidesign    function _form($raw = '', $act = 'add', $cid = NULL) {
7313011fb8bSMichael Klier        global $lang;
7323011fb8bSMichael Klier        global $conf;
7333011fb8bSMichael Klier        global $ID;
7343011fb8bSMichael Klier        global $INFO;
735f0fda08aSwikidesign
736f0fda08aSwikidesign        // not for unregistered users when guest comments aren't allowed
737f0fda08aSwikidesign        if (!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) return false;
738f0fda08aSwikidesign
7391ba72c23Swikidesign        // fill $raw with $_REQUEST['text'] if it's empty (for failed CAPTCHA check)
7401ba72c23Swikidesign        if (!$raw && ($_REQUEST['comment'] == 'show')) $raw = $_REQUEST['text'];
741f0fda08aSwikidesign        ?>
7425ef1705fSiLoveiDo
743f0fda08aSwikidesign        <div class="comment_form">
744bea69264SMichael Klier          <form id="discussion__comment_form" method="post" action="<?php echo script() ?>" accept-charset="<?php echo $lang['encoding'] ?>">
745f0fda08aSwikidesign            <div class="no">
746f0fda08aSwikidesign              <input type="hidden" name="id" value="<?php echo $ID ?>" />
74761437513Swikidesign              <input type="hidden" name="do" value="show" />
748f0fda08aSwikidesign              <input type="hidden" name="comment" value="<?php echo $act ?>" />
749ed4846efSMichael Klier              <input type="hidden" name="wikisyntaxok" id="discussion__comment_wikisyntaxok" value="<?php echo $this->getConf('wikisyntaxok') ?>" />
750530693fbSMichael Klier        <?php
751f0fda08aSwikidesign        // for adding a comment
752f0fda08aSwikidesign        if ($act == 'add') {
753f0fda08aSwikidesign        ?>
754f0fda08aSwikidesign              <input type="hidden" name="reply" value="<?php echo $cid ?>" />
755f0fda08aSwikidesign        <?php
75620c152acSMichael Klier        // for guest/adminimport: show name, e-mail and subscribe to comments fields
75720c152acSMichael Klier        if(!$_SERVER['REMOTE_USER'] or ($this->getConf('adminimport') && auth_ismanager())) {
758f0fda08aSwikidesign        ?>
759f0fda08aSwikidesign              <input type="hidden" name="user" value="<?php echo clientIP() ?>" />
760f0fda08aSwikidesign              <div class="comment_name">
761f0fda08aSwikidesign                <label class="block" for="discussion__comment_name">
762f0fda08aSwikidesign                  <span><?php echo $lang['fullname'] ?>:</span>
763e7c760b3Swikidesign                  <input type="text" class="edit" name="name" id="discussion__comment_name" size="50" tabindex="1" value="<?php echo hsc($_REQUEST['name'])?>" />
764f0fda08aSwikidesign                </label>
765f0fda08aSwikidesign              </div>
766f0fda08aSwikidesign              <div class="comment_mail">
767f0fda08aSwikidesign                <label class="block" for="discussion__comment_mail">
768f0fda08aSwikidesign                  <span><?php echo $lang['email'] ?>:</span>
7691ba72c23Swikidesign                  <input type="text" class="edit" name="mail" id="discussion__comment_mail" size="50" tabindex="2" value="<?php echo hsc($_REQUEST['mail'])?>" />
770f0fda08aSwikidesign                </label>
771f0fda08aSwikidesign              </div>
772f0fda08aSwikidesign        <?php
773f0fda08aSwikidesign        }
774f0fda08aSwikidesign
775f0fda08aSwikidesign        // allow entering an URL
776f0fda08aSwikidesign        if ($this->getConf('urlfield')) {
777f0fda08aSwikidesign        ?>
778f0fda08aSwikidesign              <div class="comment_url">
779f0fda08aSwikidesign                <label class="block" for="discussion__comment_url">
780f0fda08aSwikidesign                  <span><?php echo $this->getLang('url') ?>:</span>
781e7c760b3Swikidesign                  <input type="text" class="edit" name="url" id="discussion__comment_url" size="50" tabindex="3" value="<?php echo hsc($_REQUEST['url'])?>" />
782f0fda08aSwikidesign                </label>
783f0fda08aSwikidesign              </div>
784f0fda08aSwikidesign        <?php
785f0fda08aSwikidesign        }
786f0fda08aSwikidesign
787f0fda08aSwikidesign        // allow entering an address
788f0fda08aSwikidesign        if ($this->getConf('addressfield')) {
789f0fda08aSwikidesign        ?>
790f0fda08aSwikidesign              <div class="comment_address">
791f0fda08aSwikidesign                <label class="block" for="discussion__comment_address">
792f0fda08aSwikidesign                  <span><?php echo $this->getLang('address') ?>:</span>
793e7c760b3Swikidesign                  <input type="text" class="edit" name="address" id="discussion__comment_address" size="50" tabindex="4" value="<?php echo hsc($_REQUEST['address'])?>" />
794f0fda08aSwikidesign                </label>
795f0fda08aSwikidesign              </div>
796f0fda08aSwikidesign        <?php
797f0fda08aSwikidesign        }
798f0fda08aSwikidesign
799f0fda08aSwikidesign        // allow setting the comment date
800a1ace3c9Swikidesign        if ($this->getConf('adminimport') && (auth_ismanager())) {
801f0fda08aSwikidesign        ?>
802f0fda08aSwikidesign              <div class="comment_date">
803f0fda08aSwikidesign                <label class="block" for="discussion__comment_date">
804f0fda08aSwikidesign                  <span><?php echo $this->getLang('date') ?>:</span>
805f0fda08aSwikidesign                  <input type="text" class="edit" name="date" id="discussion__comment_date" size="50" />
806f0fda08aSwikidesign                </label>
807f0fda08aSwikidesign              </div>
808f0fda08aSwikidesign        <?php
809f0fda08aSwikidesign        }
810f0fda08aSwikidesign
811f0fda08aSwikidesign        // for saving a comment
812f0fda08aSwikidesign        } else {
813f0fda08aSwikidesign        ?>
814f0fda08aSwikidesign              <input type="hidden" name="cid" value="<?php echo $cid ?>" />
815f0fda08aSwikidesign        <?php
816f0fda08aSwikidesign        }
817f0fda08aSwikidesign        ?>
818f0fda08aSwikidesign              <div class="comment_text">
8191de52da1SMichael Klier                <div id="discussion__comment_toolbar">
8201de52da1SMichael Klier                  <?php echo $this->getLang('entercomment')?>
8211de52da1SMichael Klier                  <?php if($this->getLang('wikisyntaxok')) echo ', ' . $this->getLang('wikisyntax') . ':';?>
8221de52da1SMichael Klier                </div>
82365cdad39Swikidesign                <textarea class="edit" name="text" cols="80" rows="10" id="discussion__comment_text" tabindex="5"><?php echo formText($raw) ?></textarea>
824f0fda08aSwikidesign              </div>
825e7c760b3Swikidesign        <?php //bad and dirty event insert hook
826e7c760b3Swikidesign        $evdata = array('writable' => true);
827e7c760b3Swikidesign        trigger_event('HTML_EDITFORM_INJECTION', $evdata);
828e7c760b3Swikidesign        ?>
829bea69264SMichael Klier              <input class="button comment_submit" id="discussion__btn_submit" type="submit" name="submit" accesskey="s" value="<?php echo $lang['btn_save'] ?>" title="<?php echo $lang['btn_save']?> [S]" tabindex="7" />
830ed4846efSMichael Klier              <input class="button comment_preview" id="discussion__btn_preview" type="button" name="preview" accesskey="p" value="<?php echo $lang['btn_preview'] ?>" title="<?php echo $lang['btn_preview']?> [P]" />
8313011fb8bSMichael Klier
832189071d2SMichael Klier        <?php if((!$_SERVER['REMOTE_USER'] || $_SERVER['REMOTE_USER'] && !$conf['subscribers']) && $this->getConf('subscribe')) { ?>
8333011fb8bSMichael Klier              <div class="comment_subscribe">
83411b1339eSGina Haeussge                <input type="checkbox" id="discussion__comment_subscribe" name="subscribe" tabindex="6" />
8353011fb8bSMichael Klier                <label class="block" for="discussion__comment_subscribe">
8363011fb8bSMichael Klier                  <span><?php echo $this->getLang('subscribe') ?></span>
8373011fb8bSMichael Klier                </label>
8383011fb8bSMichael Klier              </div>
8393011fb8bSMichael Klier        <?php } ?>
8403011fb8bSMichael Klier
8413011fb8bSMichael Klier              <div class="clearer"></div>
842ed4846efSMichael Klier              <div id="discussion__comment_preview">&nbsp;</div>
843f0fda08aSwikidesign            </div>
844f0fda08aSwikidesign          </form>
845f0fda08aSwikidesign        </div>
846f0fda08aSwikidesign        <?php
847f0fda08aSwikidesign        if ($this->getConf('usecocomment')) echo $this->_coComment();
848f0fda08aSwikidesign    }
849f0fda08aSwikidesign
850f0fda08aSwikidesign    /**
851f0fda08aSwikidesign     * Adds a javascript to interact with coComments
852f0fda08aSwikidesign     */
853f0fda08aSwikidesign    function _coComment() {
8542b18adb9SMichael Klier        global $ID;
8552b18adb9SMichael Klier        global $conf;
8562b18adb9SMichael Klier        global $INFO;
857f0fda08aSwikidesign
858f0fda08aSwikidesign        $user = $_SERVER['REMOTE_USER'];
859f0fda08aSwikidesign
860f0fda08aSwikidesign        ?>
861f0fda08aSwikidesign        <script type="text/javascript"><!--//--><![CDATA[//><!--
862f0fda08aSwikidesign          var blogTool  = "DokuWiki";
863f0fda08aSwikidesign          var blogURL   = "<?php echo DOKU_URL ?>";
864f0fda08aSwikidesign          var blogTitle = "<?php echo $conf['title'] ?>";
865f0fda08aSwikidesign          var postURL   = "<?php echo wl($ID, '', true) ?>";
866f0fda08aSwikidesign          var postTitle = "<?php echo tpl_pagetitle($ID, true) ?>";
867f0fda08aSwikidesign        <?php
868f0fda08aSwikidesign        if ($user) {
869f0fda08aSwikidesign        ?>
870f0fda08aSwikidesign          var commentAuthor = "<?php echo $INFO['userinfo']['name'] ?>";
871f0fda08aSwikidesign        <?php
872f0fda08aSwikidesign        } else {
873f0fda08aSwikidesign        ?>
874f0fda08aSwikidesign          var commentAuthorFieldName = "name";
875f0fda08aSwikidesign        <?php
876f0fda08aSwikidesign        }
877f0fda08aSwikidesign        ?>
878f0fda08aSwikidesign          var commentAuthorLoggedIn = <?php echo ($user ? 'true' : 'false') ?>;
879f0fda08aSwikidesign          var commentFormID         = "discussion__comment_form";
880f0fda08aSwikidesign          var commentTextFieldName  = "text";
881f0fda08aSwikidesign          var commentButtonName     = "submit";
882f0fda08aSwikidesign          var cocomment_force       = false;
883f0fda08aSwikidesign        //--><!]]></script>
884f0fda08aSwikidesign        <script type="text/javascript" src="http://www.cocomment.com/js/cocomment.js">
885f0fda08aSwikidesign        </script>
886f0fda08aSwikidesign        <?php
887f0fda08aSwikidesign    }
888f0fda08aSwikidesign
889f0fda08aSwikidesign    /**
890f0fda08aSwikidesign     * General button function
891f0fda08aSwikidesign     */
8921e46d176Swikidesign    function _button($cid, $label, $act, $jump = false) {
893f0fda08aSwikidesign        global $ID;
8945ef1705fSiLoveiDo
8951e46d176Swikidesign        $anchor = ($jump ? '#discussion__comment_form' : '' );
896f0fda08aSwikidesign
897f0fda08aSwikidesign        ?>
8986d1f4f20SGina Haeussge        <form class="button discussion__<?php echo $act?>" method="get" action="<?php echo script().$anchor ?>">
899f0fda08aSwikidesign          <div class="no">
900f0fda08aSwikidesign            <input type="hidden" name="id" value="<?php echo $ID ?>" />
90161437513Swikidesign            <input type="hidden" name="do" value="show" />
902f0fda08aSwikidesign            <input type="hidden" name="comment" value="<?php echo $act ?>" />
903f0fda08aSwikidesign            <input type="hidden" name="cid" value="<?php echo $cid ?>" />
904f0fda08aSwikidesign            <input type="submit" value="<?php echo $label ?>" class="button" title="<?php echo $label ?>" />
905f0fda08aSwikidesign          </div>
906f0fda08aSwikidesign        </form>
907f0fda08aSwikidesign        <?php
908f0fda08aSwikidesign        return true;
909f0fda08aSwikidesign    }
910f0fda08aSwikidesign
911f0fda08aSwikidesign    /**
912f0fda08aSwikidesign     * Adds an entry to the comments changelog
913f0fda08aSwikidesign     *
914f0fda08aSwikidesign     * @author Esther Brunner <wikidesign@gmail.com>
915f0fda08aSwikidesign     * @author Ben Coburn <btcoburn@silicodon.net>
916f0fda08aSwikidesign     */
917f0fda08aSwikidesign    function _addLogEntry($date, $id, $type = 'cc', $summary = '', $extra = '') {
918f0fda08aSwikidesign        global $conf;
919f0fda08aSwikidesign
920f0fda08aSwikidesign        $changelog = $conf['metadir'].'/_comments.changes';
921f0fda08aSwikidesign
922f0fda08aSwikidesign        if(!$date) $date = time(); //use current time if none supplied
923f0fda08aSwikidesign        $remote = $_SERVER['REMOTE_ADDR'];
924f0fda08aSwikidesign        $user   = $_SERVER['REMOTE_USER'];
925f0fda08aSwikidesign
926f0fda08aSwikidesign        $strip = array("\t", "\n");
927f0fda08aSwikidesign        $logline = array(
928f0fda08aSwikidesign                'date'  => $date,
929f0fda08aSwikidesign                'ip'    => $remote,
930f0fda08aSwikidesign                'type'  => str_replace($strip, '', $type),
931f0fda08aSwikidesign                'id'    => $id,
932f0fda08aSwikidesign                'user'  => $user,
933f0fda08aSwikidesign                'sum'   => str_replace($strip, '', $summary),
934f0fda08aSwikidesign                'extra' => str_replace($strip, '', $extra)
935f0fda08aSwikidesign                );
936f0fda08aSwikidesign
937f0fda08aSwikidesign        // add changelog line
938f0fda08aSwikidesign        $logline = implode("\t", $logline)."\n";
939f0fda08aSwikidesign        io_saveFile($changelog, $logline, true); //global changelog cache
940f0fda08aSwikidesign        $this->_trimRecentCommentsLog($changelog);
94177a22ba2Swikidesign
94277a22ba2Swikidesign        // tell the indexer to re-index the page
94377a22ba2Swikidesign        @unlink(metaFN($id, '.indexed'));
944f0fda08aSwikidesign    }
945f0fda08aSwikidesign
946f0fda08aSwikidesign    /**
947f0fda08aSwikidesign     * Trims the recent comments cache to the last $conf['changes_days'] recent
948f0fda08aSwikidesign     * changes or $conf['recent'] items, which ever is larger.
949f0fda08aSwikidesign     * The trimming is only done once a day.
950f0fda08aSwikidesign     *
951f0fda08aSwikidesign     * @author Ben Coburn <btcoburn@silicodon.net>
952f0fda08aSwikidesign     */
953f0fda08aSwikidesign    function _trimRecentCommentsLog($changelog) {
954f0fda08aSwikidesign        global $conf;
955f0fda08aSwikidesign
956f0fda08aSwikidesign        if (@file_exists($changelog) &&
957f0fda08aSwikidesign                (filectime($changelog) + 86400) < time() &&
958f0fda08aSwikidesign                !@file_exists($changelog.'_tmp')) {
959f0fda08aSwikidesign
960f0fda08aSwikidesign            io_lock($changelog);
961f0fda08aSwikidesign            $lines = file($changelog);
962f0fda08aSwikidesign            if (count($lines)<$conf['recent']) {
963f0fda08aSwikidesign                // nothing to trim
964f0fda08aSwikidesign                io_unlock($changelog);
965f0fda08aSwikidesign                return true;
966f0fda08aSwikidesign            }
967f0fda08aSwikidesign
968f0fda08aSwikidesign            io_saveFile($changelog.'_tmp', '');                  // presave tmp as 2nd lock
969f0fda08aSwikidesign            $trim_time = time() - $conf['recent_days']*86400;
970f0fda08aSwikidesign            $out_lines = array();
971f0fda08aSwikidesign
972e49085a2SMichael Klier            $num = count($lines);
973e49085a2SMichael Klier            for ($i=0; $i<$num; $i++) {
974f0fda08aSwikidesign                $log = parseChangelogLine($lines[$i]);
975f0fda08aSwikidesign                if ($log === false) continue;                      // discard junk
976f0fda08aSwikidesign                if ($log['date'] < $trim_time) {
977f0fda08aSwikidesign                    $old_lines[$log['date'].".$i"] = $lines[$i];     // keep old lines for now (append .$i to prevent key collisions)
978f0fda08aSwikidesign                } else {
979f0fda08aSwikidesign                    $out_lines[$log['date'].".$i"] = $lines[$i];     // definitely keep these lines
980f0fda08aSwikidesign                }
981f0fda08aSwikidesign            }
982f0fda08aSwikidesign
983f0fda08aSwikidesign            // sort the final result, it shouldn't be necessary,
984f0fda08aSwikidesign            // however the extra robustness in making the changelog cache self-correcting is worth it
985f0fda08aSwikidesign            ksort($out_lines);
986f0fda08aSwikidesign            $extra = $conf['recent'] - count($out_lines);        // do we need extra lines do bring us up to minimum
987f0fda08aSwikidesign            if ($extra > 0) {
988f0fda08aSwikidesign                ksort($old_lines);
989f0fda08aSwikidesign                $out_lines = array_merge(array_slice($old_lines,-$extra),$out_lines);
990f0fda08aSwikidesign            }
991f0fda08aSwikidesign
992f0fda08aSwikidesign            // save trimmed changelog
993f0fda08aSwikidesign            io_saveFile($changelog.'_tmp', implode('', $out_lines));
994f0fda08aSwikidesign            @unlink($changelog);
995f0fda08aSwikidesign            if (!rename($changelog.'_tmp', $changelog)) {
996f0fda08aSwikidesign                // rename failed so try another way...
997f0fda08aSwikidesign                io_unlock($changelog);
998f0fda08aSwikidesign                io_saveFile($changelog, implode('', $out_lines));
999f0fda08aSwikidesign                @unlink($changelog.'_tmp');
1000f0fda08aSwikidesign            } else {
1001f0fda08aSwikidesign                io_unlock($changelog);
1002f0fda08aSwikidesign            }
1003f0fda08aSwikidesign            return true;
1004f0fda08aSwikidesign        }
1005f0fda08aSwikidesign    }
1006f0fda08aSwikidesign
1007f0fda08aSwikidesign    /**
1008f0fda08aSwikidesign     * Sends a notify mail on new comment
1009f0fda08aSwikidesign     *
1010f0fda08aSwikidesign     * @param  array  $comment  data array of the new comment
1011f0fda08aSwikidesign     *
1012f0fda08aSwikidesign     * @author Andreas Gohr <andi@splitbrain.org>
1013f0fda08aSwikidesign     * @author Esther Brunner <wikidesign@gmail.com>
1014f0fda08aSwikidesign     */
1015*9881d835SMichael Klier    function _notify($comment, &$subscribers) {
1016f0fda08aSwikidesign        global $conf;
1017f0fda08aSwikidesign        global $ID;
1018f0fda08aSwikidesign
1019*9881d835SMichael Klier        $notify_text = io_readfile($this->localfn('subscribermail'));
1020*9881d835SMichael Klier        $confirm_text = io_readfile($this->localfn('confirmsubscribe'));
1021*9881d835SMichael Klier        $subject_notify = '['.$conf['title'].'] '.$this->getLang('mail_newcomment');
1022*9881d835SMichael Klier        $subject_subscribe = '['.$conf['title'].'] '.$this->getLang('subscribe');
1023f0fda08aSwikidesign
10246046f25cSwikidesign        $search = array(
10256046f25cSwikidesign                '@PAGE@',
10266046f25cSwikidesign                '@TITLE@',
10276046f25cSwikidesign                '@DATE@',
10286046f25cSwikidesign                '@NAME@',
10296046f25cSwikidesign                '@TEXT@',
10308b42cefbSMichael Klier                '@COMMENTURL@',
10316046f25cSwikidesign                '@UNSUBSCRIBE@',
10326046f25cSwikidesign                '@DOKUWIKIURL@',
10336046f25cSwikidesign                );
10343011fb8bSMichael Klier
10353011fb8bSMichael Klier        // notify page subscribers
1036189071d2SMichael Klier        if ($conf['subscribers']) {
1037189071d2SMichael Klier            $list = explode(',', subscriber_addresslist($ID));
1038189071d2SMichael Klier            $to   = ($conf['notify']) ? $conf['notify'] : array_pop($list);
1039189071d2SMichael Klier            $bcc  = implode(',', $list);
10403011fb8bSMichael Klier
10416046f25cSwikidesign            $replace = array(
10426046f25cSwikidesign                    $ID,
10436046f25cSwikidesign                    $conf['title'],
1044f014bc86SMichael Klier                    strftime($conf['dformat'], $comment['date']['created']),
10456046f25cSwikidesign                    $comment['user']['name'],
10466046f25cSwikidesign                    $comment['raw'],
10471810ba9cSMichael Klier                    wl($ID, '', true) . '#comment_' . $comment['cid'],
10486046f25cSwikidesign                    wl($ID, 'do=unsubscribe', true, '&'),
10496046f25cSwikidesign                    DOKU_URL,
10506046f25cSwikidesign                    );
1051f0fda08aSwikidesign
1052*9881d835SMichael Klier                $body = str_replace($search, $replace, $notify_text);
1053*9881d835SMichael Klier                mail_send($to, $subject_notify, $body, $conf['mailfrom'], '', $bcc);
10543011fb8bSMichael Klier        }
1055f0fda08aSwikidesign
10563011fb8bSMichael Klier        // notify comment subscribers
10573011fb8bSMichael Klier        if (!empty($subscribers)) {
10583011fb8bSMichael Klier
1059*9881d835SMichael Klier            foreach($subscribers as $mail => $data) {
10603011fb8bSMichael Klier                $to = $mail;
10613011fb8bSMichael Klier
1062*9881d835SMichael Klier                if($data['active']) {
10633011fb8bSMichael Klier                    $replace = array(
10643011fb8bSMichael Klier                            $ID,
10653011fb8bSMichael Klier                            $conf['title'],
10663011fb8bSMichael Klier                            strftime($conf['dformat'], $comment['date']['created']),
10673011fb8bSMichael Klier                            $comment['user']['name'],
10683011fb8bSMichael Klier                            $comment['raw'],
10691810ba9cSMichael Klier                            wl($ID, '', true) . '#comment_' . $comment['cid'],
1070*9881d835SMichael Klier                            wl($ID, 'do=unsubscribe&hash=' . $data['hash'], true, '&'),
10713011fb8bSMichael Klier                            DOKU_URL,
10723011fb8bSMichael Klier                            );
10733011fb8bSMichael Klier
1074*9881d835SMichael Klier                    $body = str_replace($search, $replace, $notify_text);
1075*9881d835SMichael Klier                    mail_send($to, $subject_notify, $body, $conf['mailfrom']);
1076*9881d835SMichael Klier                } elseif(!$data['active'] && !$data['confirmsent']) {
1077*9881d835SMichael Klier                    $search = array(
1078*9881d835SMichael Klier                            '@PAGE@',
1079*9881d835SMichael Klier                            '@TITLE@',
1080*9881d835SMichael Klier                            '@SUBSCRIBE@',
1081*9881d835SMichael Klier                            '@DOKUWIKIURL@',
1082*9881d835SMichael Klier                            );
1083*9881d835SMichael Klier                    $replace = array(
1084*9881d835SMichael Klier                            $ID,
1085*9881d835SMichael Klier                            $conf['title'],
1086*9881d835SMichael Klier                            wl($ID, 'do=confirmsubscribe&hash=' . $data['hash'], true, '&'),
1087*9881d835SMichael Klier                            DOKU_URL,
1088*9881d835SMichael Klier                            );
1089*9881d835SMichael Klier
1090*9881d835SMichael Klier                    $body = str_replace($search, $replace, $confirm_text);
1091*9881d835SMichael Klier                    mail_send($to, $subject_subscribe, $body, $conf['mailfrom']);
1092*9881d835SMichael Klier                    $subscribers[$mail]['confirmsent'] = true;
1093*9881d835SMichael Klier                }
10943011fb8bSMichael Klier            }
10953011fb8bSMichael Klier        }
1096f0fda08aSwikidesign    }
1097f0fda08aSwikidesign
1098f0fda08aSwikidesign    /**
1099f0fda08aSwikidesign     * Counts the number of visible comments
1100f0fda08aSwikidesign     */
1101f0fda08aSwikidesign    function _count($data) {
1102f0fda08aSwikidesign        $number = 0;
1103f0fda08aSwikidesign        foreach ($data['comments'] as $cid => $comment) {
1104f0fda08aSwikidesign            if ($comment['parent']) continue;
1105f0fda08aSwikidesign            if (!$comment['show']) continue;
1106f0fda08aSwikidesign            $number++;
1107f0fda08aSwikidesign            $rids = $comment['replies'];
1108f0fda08aSwikidesign            if (count($rids)) $number = $number + $this->_countReplies($data, $rids);
1109f0fda08aSwikidesign        }
1110f0fda08aSwikidesign        return $number;
1111f0fda08aSwikidesign    }
1112f0fda08aSwikidesign
1113f0fda08aSwikidesign    function _countReplies(&$data, $rids) {
1114f0fda08aSwikidesign        $number = 0;
1115f0fda08aSwikidesign        foreach ($rids as $rid) {
11162ee3dca3Swikidesign            if (!isset($data['comments'][$rid])) continue; // reply was removed
1117f0fda08aSwikidesign            if (!$data['comments'][$rid]['show']) continue;
1118f0fda08aSwikidesign            $number++;
1119f0fda08aSwikidesign            $rids = $data['comments'][$rid]['replies'];
1120f0fda08aSwikidesign            if (count($rids)) $number = $number + $this->_countReplies($data, $rids);
1121f0fda08aSwikidesign        }
1122f0fda08aSwikidesign        return $number;
1123f0fda08aSwikidesign    }
1124f0fda08aSwikidesign
1125f0fda08aSwikidesign    /**
1126f0fda08aSwikidesign     * Renders the comment text
1127f0fda08aSwikidesign     */
1128f0fda08aSwikidesign    function _render($raw) {
1129f0fda08aSwikidesign        if ($this->getConf('wikisyntaxok')) {
1130f0fda08aSwikidesign            $xhtml = $this->render($raw);
1131f0fda08aSwikidesign        } else { // wiki syntax not allowed -> just encode special chars
1132f0fda08aSwikidesign            $xhtml = htmlspecialchars(trim($raw));
1133f0fda08aSwikidesign        }
1134f0fda08aSwikidesign        return $xhtml;
1135f0fda08aSwikidesign    }
1136f0fda08aSwikidesign
1137f0fda08aSwikidesign    /**
1138479dd10fSwikidesign     * Finds out whether there is a discussion section for the current page
1139479dd10fSwikidesign     */
1140e6a89be2Swikidesign    function _hasDiscussion(&$title) {
1141b2ac3b3bSwikidesign        global $ID;
11424a0a1bd2Swikidesign
1143479dd10fSwikidesign        $cfile = metaFN($ID, '.comments');
1144479dd10fSwikidesign
1145479dd10fSwikidesign        if (!@file_exists($cfile)) {
11462b18adb9SMichael Klier            if ($this->getConf('automatic')) {
11472b18adb9SMichael Klier                return true;
11482b18adb9SMichael Klier            } else {
11492b18adb9SMichael Klier                return false;
11502b18adb9SMichael Klier            }
1151479dd10fSwikidesign        }
1152479dd10fSwikidesign
1153479dd10fSwikidesign        $comments = unserialize(io_readFile($cfile, false));
1154479dd10fSwikidesign
115507c376bbSwikidesign        if ($comments['title']) $title = hsc($comments['title']);
1156479dd10fSwikidesign        $num = $comments['number'];
1157479dd10fSwikidesign        if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return false;
1158479dd10fSwikidesign        else return true;
1159479dd10fSwikidesign    }
1160479dd10fSwikidesign
1161479dd10fSwikidesign    /**
1162e7c760b3Swikidesign     * Creates a new thread page
1163e7c760b3Swikidesign     */
11646046f25cSwikidesign    function _newThread() {
116561437513Swikidesign        global $ID, $INFO;
1166f0fda08aSwikidesign
11671ea794e5Swikidesign        $ns    = cleanID($_REQUEST['ns']);
1168f0fda08aSwikidesign        $title = str_replace(':', '', $_REQUEST['title']);
11692e80cd5fSwikidesign        $back  = $ID;
11702e80cd5fSwikidesign        $ID    = ($ns ? $ns.':' : '').cleanID($title);
11712e80cd5fSwikidesign        $INFO  = pageinfo();
1172f0fda08aSwikidesign
1173f0fda08aSwikidesign        // check if we are allowed to create this file
11742e80cd5fSwikidesign        if ($INFO['perm'] >= AUTH_CREATE) {
1175f0fda08aSwikidesign
1176f0fda08aSwikidesign            //check if locked by anyone - if not lock for my self
11772e80cd5fSwikidesign            if ($INFO['locked']) return 'locked';
11782e80cd5fSwikidesign            else lock($ID);
1179f0fda08aSwikidesign
1180f0fda08aSwikidesign            // prepare the new thread file with default stuff
11812e80cd5fSwikidesign            if (!@file_exists($INFO['filepath'])) {
1182f0fda08aSwikidesign                global $TEXT;
1183f0fda08aSwikidesign
11842e80cd5fSwikidesign                $TEXT = pageTemplate(array(($ns ? $ns.':' : '').$title));
11851433886fSwikidesign                if (!$TEXT) {
118661437513Swikidesign                    $data = array('id' => $ID, 'ns' => $ns, 'title' => $title, 'back' => $back);
118761437513Swikidesign                    $TEXT = $this->_pageTemplate($data);
11882e80cd5fSwikidesign                }
11892e80cd5fSwikidesign                return 'preview';
1190f0fda08aSwikidesign            } else {
11912e80cd5fSwikidesign                return 'edit';
1192f0fda08aSwikidesign            }
1193f0fda08aSwikidesign        } else {
11942e80cd5fSwikidesign            return 'show';
1195f0fda08aSwikidesign        }
1196f0fda08aSwikidesign    }
1197f0fda08aSwikidesign
1198e7c760b3Swikidesign    /**
119961437513Swikidesign     * Adapted version of pageTemplate() function
120061437513Swikidesign     */
120161437513Swikidesign    function _pageTemplate($data) {
120261437513Swikidesign        global $conf, $INFO;
120361437513Swikidesign
120461437513Swikidesign        $id   = $data['id'];
120561437513Swikidesign        $user = $_SERVER['REMOTE_USER'];
120661437513Swikidesign        $tpl  = io_readFile(DOKU_PLUGIN.'discussion/_template.txt');
120761437513Swikidesign
120861437513Swikidesign        // standard replacements
120961437513Swikidesign        $replace = array(
121061437513Swikidesign                '@NS@'   => $data['ns'],
121161437513Swikidesign                '@PAGE@' => strtr(noNS($id),'_',' '),
121261437513Swikidesign                '@USER@' => $user,
121361437513Swikidesign                '@NAME@' => $INFO['userinfo']['name'],
121461437513Swikidesign                '@MAIL@' => $INFO['userinfo']['mail'],
1215f014bc86SMichael Klier                '@DATE@' => strftime($conf['dformat']),
121661437513Swikidesign                );
121761437513Swikidesign
121861437513Swikidesign        // additional replacements
121961437513Swikidesign        $replace['@BACK@']  = $data['back'];
122061437513Swikidesign        $replace['@TITLE@'] = $data['title'];
122161437513Swikidesign
122261437513Swikidesign        // avatar if useavatar and avatar plugin available
122361437513Swikidesign        if ($this->getConf('useavatar')
122461437513Swikidesign                && (@file_exists(DOKU_PLUGIN.'avatar/syntax.php'))
122561437513Swikidesign                && (!plugin_isdisabled('avatar'))) {
122661437513Swikidesign            $replace['@AVATAR@'] = '{{avatar>'.$user.' }} ';
122761437513Swikidesign        } else {
122861437513Swikidesign            $replace['@AVATAR@'] = '';
122961437513Swikidesign        }
123061437513Swikidesign
123161437513Swikidesign        // tag if tag plugin is available
123261437513Swikidesign        if ((@file_exists(DOKU_PLUGIN.'tag/syntax/tag.php'))
123361437513Swikidesign                && (!plugin_isdisabled('tag'))) {
123461437513Swikidesign            $replace['@TAG@'] = "\n\n{{tag>}}";
123561437513Swikidesign        } else {
123661437513Swikidesign            $replace['@TAG@'] = '';
123761437513Swikidesign        }
123861437513Swikidesign
123961437513Swikidesign        // do the replace
124061437513Swikidesign        $tpl = str_replace(array_keys($replace), array_values($replace), $tpl);
124161437513Swikidesign        return $tpl;
124261437513Swikidesign    }
124361437513Swikidesign
124461437513Swikidesign    /**
1245e7c760b3Swikidesign     * Checks if the CAPTCHA string submitted is valid
1246e7c760b3Swikidesign     *
1247e7c760b3Swikidesign     * @author     Andreas Gohr <gohr@cosmocode.de>
1248e7c760b3Swikidesign     * @adaption   Esther Brunner <wikidesign@gmail.com>
1249e7c760b3Swikidesign     */
12506046f25cSwikidesign    function _captchaCheck() {
1251be1d1f2fSGina Haeussge        if (plugin_isdisabled('captcha') || (!$captcha = plugin_load('helper', 'captcha')))
1252be1d1f2fSGina Haeussge            return; // CAPTCHA is disabled or not available
1253e7c760b3Swikidesign
1254d1c29589Swikidesign        // do nothing if logged in user and no CAPTCHA required
1255d1c29589Swikidesign        if (!$captcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return;
1256d1c29589Swikidesign
1257e7c760b3Swikidesign        // compare provided string with decrypted captcha
1258e7c760b3Swikidesign        $rand = PMA_blowfish_decrypt($_REQUEST['plugin__captcha_secret'], auth_cookiesalt());
1259e7c760b3Swikidesign        $code = $captcha->_generateCAPTCHA($captcha->_fixedIdent(), $rand);
1260e7c760b3Swikidesign
1261e7c760b3Swikidesign        if (!$_REQUEST['plugin__captcha_secret'] ||
1262e7c760b3Swikidesign                !$_REQUEST['plugin__captcha'] ||
1263e7c760b3Swikidesign                strtoupper($_REQUEST['plugin__captcha']) != $code) {
1264e7c760b3Swikidesign
1265e7c760b3Swikidesign            // CAPTCHA test failed! Continue to edit instead of saving
1266e7c760b3Swikidesign            msg($captcha->getLang('testfailed'), -1);
1267e7c760b3Swikidesign            if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit';
1268e7c760b3Swikidesign            elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show';
1269e7c760b3Swikidesign        }
1270e7c760b3Swikidesign        // if we arrive here it was a valid save
1271e7c760b3Swikidesign    }
1272e7c760b3Swikidesign
1273a1ca9e44Swikidesign    /**
1274a1ca9e44Swikidesign     * Adds the comments to the index
1275a1ca9e44Swikidesign     */
127661437513Swikidesign    function idx_add_discussion(&$event, $param) {
1277a1ca9e44Swikidesign
1278a1ca9e44Swikidesign        // get .comments meta file name
1279a1ca9e44Swikidesign        $file = metaFN($event->data[0], '.comments');
1280a1ca9e44Swikidesign
1281a1ca9e44Swikidesign        if (@file_exists($file)) $data = unserialize(io_readFile($file, false));
1282a1ca9e44Swikidesign        if ((!$data['status']) || ($data['number'] == 0)) return; // comments are turned off
1283a1ca9e44Swikidesign
1284a1ca9e44Swikidesign        // now add the comments
1285a1ca9e44Swikidesign        if (isset($data['comments'])) {
1286a1ca9e44Swikidesign            foreach ($data['comments'] as $key => $value) {
128777a22ba2Swikidesign                $event->data[1] .= $this->_addCommentWords($key, $data);
1288a1ca9e44Swikidesign            }
1289a1ca9e44Swikidesign        }
1290a1ca9e44Swikidesign    }
1291a1ca9e44Swikidesign
1292a1ca9e44Swikidesign    /**
1293a1ca9e44Swikidesign     * Adds the words of a given comment to the index
1294a1ca9e44Swikidesign     */
1295efbe59d0Swikidesign    function _addCommentWords($cid, &$data, $parent = '') {
1296a1ca9e44Swikidesign
1297efbe59d0Swikidesign        if (!isset($data['comments'][$cid])) return ''; // comment was removed
1298a1ca9e44Swikidesign        $comment = $data['comments'][$cid];
1299a1ca9e44Swikidesign
1300efbe59d0Swikidesign        if (!is_array($comment)) return '';             // corrupt datatype
1301efbe59d0Swikidesign        if ($comment['parent'] != $parent) return '';   // reply to an other comment
1302efbe59d0Swikidesign        if (!$comment['show']) return '';               // hidden comment
1303a1ca9e44Swikidesign
1304efbe59d0Swikidesign        $text = $comment['raw'];                        // we only add the raw comment text
1305efbe59d0Swikidesign        if (is_array($comment['replies'])) {             // and the replies
1306efbe59d0Swikidesign            foreach ($comment['replies'] as $rid) {
1307efbe59d0Swikidesign                $text .= $this->_addCommentWords($rid, $data, $cid);
1308a1ca9e44Swikidesign            }
1309a1ca9e44Swikidesign        }
1310efbe59d0Swikidesign        return ' '.$text;
1311efbe59d0Swikidesign    }
1312a10b5c98SMichael Klier
1313a10b5c98SMichael Klier    /**
1314a10b5c98SMichael Klier     * Only allow http(s) URLs and append http:// to URLs if needed
1315a10b5c98SMichael Klier     */
1316a10b5c98SMichael Klier    function _checkURL($url) {
1317a10b5c98SMichael Klier        if(preg_match("#^http://|^https://#", $url)) {
1318a10b5c98SMichael Klier            return hsc($url);
1319a10b5c98SMichael Klier        } elseif(substr($url, 0, 4) == 'www.') {
1320a10b5c98SMichael Klier            return hsc('http://' . $url);
1321a10b5c98SMichael Klier        } else {
1322a10b5c98SMichael Klier            return '';
1323a10b5c98SMichael Klier        }
1324a10b5c98SMichael Klier    }
1325f0fda08aSwikidesign}
132631aab30eSGina Haeussge
132731aab30eSGina Haeussgefunction _sortCallback($a, $b) {
132831aab30eSGina Haeussge    if (is_array($a['date'])) { // new format
132931aab30eSGina Haeussge        $createdA  = $a['date']['created'];
133031aab30eSGina Haeussge    } else {                         // old format
133131aab30eSGina Haeussge        $createdA  = $a['date'];
133231aab30eSGina Haeussge    }
133331aab30eSGina Haeussge
133431aab30eSGina Haeussge    if (is_array($b['date'])) { // new format
133531aab30eSGina Haeussge        $createdB  = $b['date']['created'];
133631aab30eSGina Haeussge    } else {                         // old format
133731aab30eSGina Haeussge        $createdB  = $b['date'];
133831aab30eSGina Haeussge    }
133931aab30eSGina Haeussge
134031aab30eSGina Haeussge    if ($createdA == $createdB)
134131aab30eSGina Haeussge        return 0;
134231aab30eSGina Haeussge    else
134331aab30eSGina Haeussge        return ($createdA < $createdB) ? -1 : 1;
134431aab30eSGina Haeussge}
134531aab30eSGina Haeussge
1346530693fbSMichael Klier// vim:ts=4:sw=4:et:enc=utf-8:
1347