xref: /plugin/discussion/action.php (revision 5ef1705f41041912b7883d46c49e8adc3908c1e7)
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
10f0fda08aSwikidesignif (!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/');
11f0fda08aSwikidesignrequire_once(DOKU_PLUGIN.'action.php');
12f0fda08aSwikidesign
13f0fda08aSwikidesignif (!defined('NL')) define('NL',"\n");
14f0fda08aSwikidesign
15f0fda08aSwikidesignclass action_plugin_discussion extends DokuWiki_Action_Plugin{
16f0fda08aSwikidesign
17f0fda08aSwikidesign  /**
18f0fda08aSwikidesign   * Return some info
19f0fda08aSwikidesign   */
20f0fda08aSwikidesign  function getInfo(){
21f0fda08aSwikidesign    return array(
22f0fda08aSwikidesign      'author' => 'Esther Brunner',
23f0fda08aSwikidesign      'email'  => 'wikidesign@gmail.com',
24830f19a9Swikidesign      'date'   => '2007-01-05',
25f0fda08aSwikidesign      'name'   => 'Discussion Plugin',
26f0fda08aSwikidesign      'desc'   => 'Enables discussion features',
27decf3d57Swikidesign      'url'    => 'http://www.wikidesign.ch/en/plugin/discussion/start',
28f0fda08aSwikidesign    );
29f0fda08aSwikidesign  }
30f0fda08aSwikidesign
31f0fda08aSwikidesign  /**
32f0fda08aSwikidesign   * Register the eventhandlers
33f0fda08aSwikidesign   */
34f0fda08aSwikidesign  function register(&$contr){
35f0fda08aSwikidesign    $contr->register_hook(
36f0fda08aSwikidesign      'ACTION_ACT_PREPROCESS',
37f0fda08aSwikidesign      'BEFORE',
38f0fda08aSwikidesign      $this,
39f0fda08aSwikidesign      'handle_act_preprocess',
40f0fda08aSwikidesign      array()
41f0fda08aSwikidesign    );
42f0fda08aSwikidesign    $contr->register_hook(
43f0fda08aSwikidesign      'TPL_ACT_RENDER',
44f0fda08aSwikidesign      'AFTER',
45f0fda08aSwikidesign      $this,
46f0fda08aSwikidesign      'comments',
47f0fda08aSwikidesign      array()
48f0fda08aSwikidesign    );
49479dd10fSwikidesign    $contr->register_hook(
50479dd10fSwikidesign      'RENDERER_CONTENT_POSTPROCESS',
51479dd10fSwikidesign      'AFTER',
52479dd10fSwikidesign      $this,
53479dd10fSwikidesign      'add_toc_item',
54479dd10fSwikidesign      array()
55479dd10fSwikidesign    );
56f0fda08aSwikidesign  }
57f0fda08aSwikidesign
58f0fda08aSwikidesign  /**
59f0fda08aSwikidesign   * Main function; dispatches the comment actions
60f0fda08aSwikidesign   */
61f0fda08aSwikidesign  function comments(&$event, $param){
62*5ef1705fSiLoveiDo    global $ADMDISCUSSION;
63*5ef1705fSiLoveiDo
64*5ef1705fSiLoveiDo    if (($event->data != 'admin') && ($event->data != 'show')) return; // nothing to do for us
65*5ef1705fSiLoveiDo
66*5ef1705fSiLoveiDo    if(isset($ADMDISCUSSION['breakaction'])) return;
67f0fda08aSwikidesign
68f0fda08aSwikidesign    $cid  = $_REQUEST['cid'];
69f0fda08aSwikidesign
70f0fda08aSwikidesign    switch ($_REQUEST['comment']){
71f0fda08aSwikidesign
72f0fda08aSwikidesign      case 'add':
73f0fda08aSwikidesign        $comment = array(
746046f25cSwikidesign          'user'    => array(
756046f25cSwikidesign            'id'      => hsc($_REQUEST['user']),
766046f25cSwikidesign            'name'    => hsc($_REQUEST['name']),
776046f25cSwikidesign            'mail'    => hsc($_REQUEST['mail']),
786046f25cSwikidesign            'url'     => hsc($_REQUEST['url']),
796046f25cSwikidesign            'address' => hsc($_REQUEST['address'])),
806046f25cSwikidesign          'date'    => array('created' => $_REQUEST['date']),
81f0fda08aSwikidesign          'raw'     => cleanText($_REQUEST['text'])
82f0fda08aSwikidesign        );
83f0fda08aSwikidesign        $repl = $_REQUEST['reply'];
84f0fda08aSwikidesign        $this->_add($comment, $repl);
85f0fda08aSwikidesign        break;
86f0fda08aSwikidesign
87f0fda08aSwikidesign      case 'edit':
88f0fda08aSwikidesign        $this->_show(NULL, $cid);
89f0fda08aSwikidesign        break;
90f0fda08aSwikidesign
91f0fda08aSwikidesign      case 'save':
92f0fda08aSwikidesign        $raw  = cleanText($_REQUEST['text']);
93f0fda08aSwikidesign        $this->_save($cid, $raw);
94f0fda08aSwikidesign        break;
95f0fda08aSwikidesign
961e46d176Swikidesign      case 'delete':
971e46d176Swikidesign        $this->_save($cid, '');
982ee3dca3Swikidesign        break;
991e46d176Swikidesign
100f0fda08aSwikidesign      case 'toogle':
101f0fda08aSwikidesign        $this->_save($cid, '', true);
102f0fda08aSwikidesign        break;
103f0fda08aSwikidesign
104f0fda08aSwikidesign      default: // 'show' => $this->_show(), 'reply' => $this->_show($cid)
105f0fda08aSwikidesign        $this->_show($cid);
106f0fda08aSwikidesign    }
107f0fda08aSwikidesign  }
108f0fda08aSwikidesign
109f0fda08aSwikidesign  /**
110f0fda08aSwikidesign   * Shows all comments of the current page
111f0fda08aSwikidesign   */
112f0fda08aSwikidesign  function _show($reply = NULL, $edit = NULL){
113434f5172SiLoveiDo    global $ID, $INFO , $ADMDISCUSSION;
114f0fda08aSwikidesign
115479dd10fSwikidesign    // get .comments meta file name
116f0fda08aSwikidesign    $file = metaFN($ID, '.comments');
117f0fda08aSwikidesign
118479dd10fSwikidesign    if (!@file_exists($file)){
119479dd10fSwikidesign      // create .comments meta file if automatic setting is switched on
1207fec4787Swikidesign      if ($this->getConf('automatic') && $INFO['exists']){
121479dd10fSwikidesign        $data = array('status' => 1, 'number' => 0);
122479dd10fSwikidesign        io_saveFile($file, serialize($data));
123479dd10fSwikidesign      }
124479dd10fSwikidesign    } else { // load data
125f0fda08aSwikidesign      $data = unserialize(io_readFile($file, false));
126479dd10fSwikidesign    }
127f0fda08aSwikidesign
128479dd10fSwikidesign    if (!$data['status']) return false; // comments are turned off
129f0fda08aSwikidesign
130f0fda08aSwikidesign    // section title
131434f5172SiLoveiDo    $title = $this->getLang('discussion').((isset($ADMDISCUSSION['page']))?$ADMDISCUSSION['page']:'');
132decf3d57Swikidesign    echo '<div class="comment_wrapper">';
133479dd10fSwikidesign    echo '<h2><a name="discussion__section" id="discussion__section">'.$title.'</a></h2>';
134830f19a9Swikidesign    echo '<div class="level2 hfeed">';
135f0fda08aSwikidesign
136f0fda08aSwikidesign    // now display the comments
137f0fda08aSwikidesign    if (isset($data['comments'])){
138f0fda08aSwikidesign      foreach ($data['comments'] as $key => $value){
139f0fda08aSwikidesign        if ($key == $edit) $this->_form($value['raw'], 'save', $edit); // edit form
140f0fda08aSwikidesign        else $this->_print($key, $data, '', $reply);
141f0fda08aSwikidesign      }
142f0fda08aSwikidesign    }
143f0fda08aSwikidesign
144f0fda08aSwikidesign    // comment form
145f0fda08aSwikidesign    if (($data['status'] == 1) && !$reply && !$edit) $this->_form('');
146f0fda08aSwikidesign
147decf3d57Swikidesign    echo '</div>'; // level2
148decf3d57Swikidesign    echo '</div>'; // comment_wrapper
149f0fda08aSwikidesign
150f0fda08aSwikidesign    return true;
151f0fda08aSwikidesign  }
152f0fda08aSwikidesign
153f0fda08aSwikidesign  /**
154f0fda08aSwikidesign   * Adds a new comment and then displays all comments
155f0fda08aSwikidesign   */
156f0fda08aSwikidesign  function _add($comment, $parent){
157f0fda08aSwikidesign    global $ID;
158f0fda08aSwikidesign    global $TEXT;
159*5ef1705fSiLoveiDo    global $ADMDISCUSSION;
160f0fda08aSwikidesign
161f0fda08aSwikidesign    $otxt = $TEXT; // set $TEXT to comment text for wordblock check
162f0fda08aSwikidesign    $TEXT = $comment['raw'];
163f0fda08aSwikidesign
164f0fda08aSwikidesign    // spamcheck against the DokuWiki blacklist
165f0fda08aSwikidesign    if (checkwordblock()){
166f0fda08aSwikidesign      msg($this->getLang('wordblock'), -1);
167f0fda08aSwikidesign      $this->_show();
168f0fda08aSwikidesign      return false;
169f0fda08aSwikidesign    }
170f0fda08aSwikidesign
171f0fda08aSwikidesign    $TEXT = $otxt; // restore global $TEXT
172f0fda08aSwikidesign
173f0fda08aSwikidesign    // get discussion meta file name
174f0fda08aSwikidesign    $file = metaFN($ID, '.comments');
175f0fda08aSwikidesign
176f0fda08aSwikidesign    $data = array();
177f0fda08aSwikidesign    $data = unserialize(io_readFile($file, false));
178f0fda08aSwikidesign
179f0fda08aSwikidesign    if ($data['status'] != 1) return false;                // comments off or closed
180f0fda08aSwikidesign    if ((!$this->getConf('allowguests'))
1816046f25cSwikidesign      && ($comment['user']['id'] != $_SERVER['REMOTE_USER']))
182f0fda08aSwikidesign      return false;                                        // guest comments not allowed
183f0fda08aSwikidesign
1846046f25cSwikidesign    if ($comment['date']['created']) $date = strtotime($comment['date']['created']);
185f0fda08aSwikidesign    else $date = time();
186f0fda08aSwikidesign    if ($date == -1) $date = time();
1876046f25cSwikidesign    $cid  = md5($comment['user']['id'].$date);             // create a unique id
188f0fda08aSwikidesign
189f0fda08aSwikidesign    if (!is_array($data['comments'][$parent])) $parent = NULL; // invalid parent comment
190f0fda08aSwikidesign
191f0fda08aSwikidesign    // render the comment
192f0fda08aSwikidesign    $xhtml = $this->_render($comment['raw']);
193f0fda08aSwikidesign
194f0fda08aSwikidesign    // fill in the new comment
195f0fda08aSwikidesign    $data['comments'][$cid] = array(
1966046f25cSwikidesign      'user'    => $comment['user'],
1976046f25cSwikidesign      'date'    => array('created' => $date),
198f0fda08aSwikidesign      'show'    => true,
1996046f25cSwikidesign      'raw'     => $comment['raw'],
200f0fda08aSwikidesign      'xhtml'   => $xhtml,
201f0fda08aSwikidesign      'parent'  => $parent,
202f0fda08aSwikidesign      'replies' => array()
203f0fda08aSwikidesign    );
204f0fda08aSwikidesign
205f0fda08aSwikidesign    // update parent comment
206f0fda08aSwikidesign    if ($parent) $data['comments'][$parent]['replies'][] = $cid;
207f0fda08aSwikidesign
208f0fda08aSwikidesign    // update the number of comments
209f0fda08aSwikidesign    $data['number']++;
210f0fda08aSwikidesign
211f0fda08aSwikidesign    // save the comment metadata file
212f0fda08aSwikidesign    io_saveFile($file, serialize($data));
213f0fda08aSwikidesign    $this->_addLogEntry($date, $ID, 'cc', '', $cid);
214f0fda08aSwikidesign
215f0fda08aSwikidesign    // notify subscribers of the page
216f0fda08aSwikidesign    $this->_notify($data['comments'][$cid]);
217f0fda08aSwikidesign
218*5ef1705fSiLoveiDo    if (!isset($ADMDISCUSSION['page']))
219f0fda08aSwikidesign    	$this->_show();
220f0fda08aSwikidesign    return true;
221f0fda08aSwikidesign  }
222f0fda08aSwikidesign
223f0fda08aSwikidesign  /**
224f0fda08aSwikidesign   * Saves the comment with the given ID and then displays all comments
225f0fda08aSwikidesign   */
226f0fda08aSwikidesign  function _save($cid, $raw, $toogle = false){
227f0fda08aSwikidesign    global $ID;
228f0fda08aSwikidesign    global $INFO;
229*5ef1705fSiLoveiDo    global $ADMDISCUSSION;
230f0fda08aSwikidesign
2312ee3dca3Swikidesign    if ($raw){
2322ee3dca3Swikidesign      global $TEXT;
2332ee3dca3Swikidesign
234f0fda08aSwikidesign      $otxt = $TEXT; // set $TEXT to comment text for wordblock check
235f0fda08aSwikidesign      $TEXT = $raw;
236f0fda08aSwikidesign
237f0fda08aSwikidesign      // spamcheck against the DokuWiki blacklist
238f0fda08aSwikidesign      if (checkwordblock()){
239f0fda08aSwikidesign        msg($this->getLang('wordblock'), -1);
240f0fda08aSwikidesign        $this->_show();
241f0fda08aSwikidesign        return false;
242f0fda08aSwikidesign      }
243f0fda08aSwikidesign
244f0fda08aSwikidesign      $TEXT = $otxt; // restore global $TEXT
2452ee3dca3Swikidesign    }
246f0fda08aSwikidesign
247f0fda08aSwikidesign    // get discussion meta file name
248f0fda08aSwikidesign    $file = metaFN($ID, '.comments');
249f0fda08aSwikidesign
250f0fda08aSwikidesign    $data = array();
251f0fda08aSwikidesign    $data = unserialize(io_readFile($file, false));
252f0fda08aSwikidesign
2536046f25cSwikidesign    if (is_array($data['comments'][$cid]['user'])){
2546046f25cSwikidesign      $user    = $data['comments'][$cid]['user']['id'];
2556046f25cSwikidesign      $convert = false;
2566046f25cSwikidesign    } else {
2576046f25cSwikidesign      $user    = $data['comments'][$cid]['user'];
2586046f25cSwikidesign      $convert = true;
2596046f25cSwikidesign    }
2606046f25cSwikidesign
261f0fda08aSwikidesign    // someone else was trying to edit our comment -> abort
2626046f25cSwikidesign    if (($user != $_SERVER['REMOTE_USER']) && ($INFO['perm'] != AUTH_ADMIN)) return false;
263f0fda08aSwikidesign
264f0fda08aSwikidesign    $date = time();
265f0fda08aSwikidesign
2666046f25cSwikidesign    // need to convert to new format?
2676046f25cSwikidesign    if ($convert){
2686046f25cSwikidesign      $data['comments'][$cid]['user'] = array(
2696046f25cSwikidesign        'id'      => $user,
2706046f25cSwikidesign        'name'    => $data['comments'][$cid]['name'],
2716046f25cSwikidesign        'mail'    => $data['comments'][$cid]['mail'],
2726046f25cSwikidesign        'url'     => $data['comments'][$cid]['url'],
2736046f25cSwikidesign        'address' => $data['comments'][$cid]['address'],
2746046f25cSwikidesign      );
2756046f25cSwikidesign      $data['comments'][$cid]['date'] = array(
2766046f25cSwikidesign        'created' => $data['comments'][$cid]['date']
2776046f25cSwikidesign      );
2786046f25cSwikidesign    }
2796046f25cSwikidesign
280f0fda08aSwikidesign    if ($toogle){     // toogle visibility
281f0fda08aSwikidesign      $now = $data['comments'][$cid]['show'];
282f0fda08aSwikidesign      $data['comments'][$cid]['show'] = !$now;
283f0fda08aSwikidesign      $data['number'] = $this->_count($data);
284f0fda08aSwikidesign
285f0fda08aSwikidesign      $type = ($data['comments'][$cid]['show'] ? 'sc' : 'hc');
286f0fda08aSwikidesign
287f0fda08aSwikidesign    } elseif (!$raw){ // remove the comment
288f0fda08aSwikidesign      unset($data['comments'][$cid]);
289f0fda08aSwikidesign      $data['number'] = $this->_count($data);
290f0fda08aSwikidesign
291f0fda08aSwikidesign      $type = 'dc';
292f0fda08aSwikidesign
293f0fda08aSwikidesign    } else {          // save changed comment
294f0fda08aSwikidesign      $xhtml = $this->_render($raw);
295f0fda08aSwikidesign
296f0fda08aSwikidesign      // now change the comment's content
2976046f25cSwikidesign      $data['comments'][$cid]['date']['modified'] = $date;
2986046f25cSwikidesign      $data['comments'][$cid]['raw']              = $raw;
299f0fda08aSwikidesign      $data['comments'][$cid]['xhtml']            = $xhtml;
300f0fda08aSwikidesign
301f0fda08aSwikidesign      $type = 'ec';
302f0fda08aSwikidesign    }
303f0fda08aSwikidesign
304f0fda08aSwikidesign    // save the comment metadata file
305*5ef1705fSiLoveiDo
306f0fda08aSwikidesign    io_saveFile($file, serialize($data));
307f0fda08aSwikidesign    $this->_addLogEntry($date, $ID, $type, '', $cid);
308f0fda08aSwikidesign
309*5ef1705fSiLoveiDo    if (!isset($ADMDISCUSSION['page']))
310f0fda08aSwikidesign    	$this->_show();
311f0fda08aSwikidesign    return true;
312f0fda08aSwikidesign  }
313f0fda08aSwikidesign
314f0fda08aSwikidesign  /**
315f0fda08aSwikidesign   * Prints an individual comment
316f0fda08aSwikidesign   */
317f0fda08aSwikidesign  function _print($cid, &$data, $parent = '', $reply = '', $visible = true){
318f0fda08aSwikidesign    global $conf;
319f0fda08aSwikidesign    global $lang;
320f0fda08aSwikidesign    global $ID;
321f0fda08aSwikidesign    global $INFO;
322f0fda08aSwikidesign
3232ee3dca3Swikidesign    if (!isset($data['comments'][$cid])) return false; // comment was removed
324f0fda08aSwikidesign    $comment = $data['comments'][$cid];
325f0fda08aSwikidesign
326f0fda08aSwikidesign    if (!is_array($comment)) return false;             // corrupt datatype
327f0fda08aSwikidesign
328f0fda08aSwikidesign    if ($comment['parent'] != $parent) return true;    // reply to an other comment
329f0fda08aSwikidesign
330f0fda08aSwikidesign    if (!$comment['show']){                            // comment hidden
331f0fda08aSwikidesign      if ($INFO['perm'] == AUTH_ADMIN) echo '<div class="comment_hidden">'.NL;
332f0fda08aSwikidesign      else return true;
333f0fda08aSwikidesign    }
334f0fda08aSwikidesign
335f0fda08aSwikidesign    // comment head with date and user data
336830f19a9Swikidesign    echo '<div class="hentry"><div class="comment_head">'.NL.
337830f19a9Swikidesign      '<a name="comment__'.$cid.'" id="comment__'.$cid.'"></a>'.NL.
338830f19a9Swikidesign      '<span class="vcard author">';
339f0fda08aSwikidesign
3406046f25cSwikidesign    // prepare variables
3416046f25cSwikidesign    if (is_array($comment['user'])){ // new format
3426046f25cSwikidesign      $user    = $comment['user']['id'];
3436046f25cSwikidesign      $name    = $comment['user']['name'];
3446046f25cSwikidesign      $mail    = $comment['user']['mail'];
3456046f25cSwikidesign      $url     = $comment['user']['url'];
3466046f25cSwikidesign      $address = $comment['user']['address'];
3476046f25cSwikidesign    } else {                         // old format
3486046f25cSwikidesign      $user    = $comment['user'];
3496046f25cSwikidesign      $name    = $comment['name'];
3506046f25cSwikidesign      $mail    = $comment['mail'];
3516046f25cSwikidesign      $url     = $comment['url'];
3526046f25cSwikidesign      $address = $comment['address'];
3536046f25cSwikidesign    }
3546046f25cSwikidesign    if (is_array($comment['date'])){ // new format
3556046f25cSwikidesign      $created  = $comment['date']['created'];
3566046f25cSwikidesign      $modified = $comment['date']['modified'];
3576046f25cSwikidesign    } else {                         // old format
3586046f25cSwikidesign      $created  = $comment['date'];
3596046f25cSwikidesign      $modified = $comment['edited'];
3606046f25cSwikidesign    }
3616046f25cSwikidesign
362f0fda08aSwikidesign    // show gravatar image
363f0fda08aSwikidesign    if ($this->getConf('usegravatar')){
364f0fda08aSwikidesign      $default = DOKU_URL.'lib/plugins/discussion/images/default.gif';
365f0fda08aSwikidesign      $size    = $this->getConf('gravatar_size');
3666046f25cSwikidesign      if ($mail) $src = ml('http://www.gravatar.com/avatar.php?'.
3676046f25cSwikidesign        'gravatar_id='.md5($mail).
368f0fda08aSwikidesign        '&default='.urlencode($default).
369f0fda08aSwikidesign        '&size='.$size.
3700c7c2bf6Swikidesign        '&rating='.$this->getConf('gravatar_rating').
3710c7c2bf6Swikidesign        '&.jpg', 'cache=recache');
372f0fda08aSwikidesign      else $src = $default;
3736046f25cSwikidesign      $title = ($name ? $name : obfuscate($mail));
374434f5172SiLoveiDo      echo '<ADMDISCUSSIONimg src="'.$src.'" class="medialeft photo" title="'.$title.'"'.
375f0fda08aSwikidesign        ' alt="'.$title.'" width="'.$size.'" height="'.$size.'" />'.NL;
3765f87b5b6Swikidesign      $style = ' style="margin-left: '.($size + 14).'px;"';
3775f87b5b6Swikidesign    } else {
3785f87b5b6Swikidesign      $style = ' style="margin-left: 20px;"';
379f0fda08aSwikidesign    }
380f0fda08aSwikidesign
3816046f25cSwikidesign    if ($this->getConf('linkemail') && $mail){
3826046f25cSwikidesign      echo $this->email($mail, $name, 'email fn');
3836046f25cSwikidesign    } elseif ($url){
3846046f25cSwikidesign      echo $this->external_link($url, $name, 'urlextern url fn');
385f0fda08aSwikidesign    } else {
3866046f25cSwikidesign      echo '<span class="fn">'.$name.'</span>';
387f0fda08aSwikidesign    }
3886046f25cSwikidesign    if ($address) echo ', <span class="adr">'.$address.'</span>';
3896046f25cSwikidesign    echo '</span>, <abbr class="published" title="'.gmdate('Y-m-d\TH:i:s\Z', $created).
3906046f25cSwikidesign      '">'.date($conf['dformat'], $created).'</abbr>';
391830f19a9Swikidesign    if ($comment['edited']) echo ' (<abbr class="updated" title="'.
3926046f25cSwikidesign      gmdate('Y-m-d\TH:i:s\Z', $modified).'">'.date($conf['dformat'], $modified).
3936046f25cSwikidesign      '</abbr>)';
3946046f25cSwikidesign    echo ':'.NL.'</div>'.NL; // class="comment_head"
395f0fda08aSwikidesign
396f0fda08aSwikidesign    // main comment content
397830f19a9Swikidesign    echo '<div class="comment_body entry-content"'.
398830f19a9Swikidesign      ($this->getConf('usegravatar') ? $style : '').'>'.NL.
399830f19a9Swikidesign      $comment['xhtml'].NL.
4006046f25cSwikidesign      '</div>'.NL.'</div>'.NL; // class="comment_body" and class="hentry"
401f0fda08aSwikidesign
402f0fda08aSwikidesign    if ($visible){
403f0fda08aSwikidesign      // show hide/show toogle button?
404f0fda08aSwikidesign      echo '<div class="comment_buttons">'.NL;
405f0fda08aSwikidesign      if ($INFO['perm'] == AUTH_ADMIN){
406f0fda08aSwikidesign        if (!$comment['show']) $label = $this->getLang('btn_show');
407f0fda08aSwikidesign        else $label = $this->getLang('btn_hide');
408f0fda08aSwikidesign
409f0fda08aSwikidesign        $this->_button($cid, $label, 'toogle');
410f0fda08aSwikidesign      }
411f0fda08aSwikidesign
412f0fda08aSwikidesign      // show reply button?
413f1c4aa1aSwikidesign      if (($data['status'] == 1) && !$reply && $comment['show']
414f1c4aa1aSwikidesign        && ($this->getConf('allowguests') || $_SERVER['REMOTE_USER']))
4151e46d176Swikidesign        $this->_button($cid, $this->getLang('btn_reply'), 'reply', true);
416f0fda08aSwikidesign
4171e46d176Swikidesign      // show edit and delete button?
4186046f25cSwikidesign      if ((($user == $_SERVER['REMOTE_USER']) && ($user != ''))
419f0fda08aSwikidesign        || ($INFO['perm'] == AUTH_ADMIN))
4201e46d176Swikidesign        $this->_button($cid, $lang['btn_secedit'], 'edit', true);
4211e46d176Swikidesign      if ($INFO['perm'] == AUTH_ADMIN)
4221e46d176Swikidesign        $this->_button($cid, $lang['btn_delete'], 'delete');
423f0fda08aSwikidesign      echo '</div>'.NL; // class="comment_buttons"
4246046f25cSwikidesign      echo '<div class="comment_line" '.
4256046f25cSwikidesign        ($this->getConf('usegravatar') ? $style : '').'>&nbsp;</div>'.NL;
426f0fda08aSwikidesign    }
427f0fda08aSwikidesign
428f0fda08aSwikidesign    // replies to this comment entry?
429f0fda08aSwikidesign    if (count($comment['replies'])){
4305f87b5b6Swikidesign      echo '<div class="comment_replies"'.$style.'>'.NL;
431f0fda08aSwikidesign      $visible = ($comment['show'] && $visible);
432f0fda08aSwikidesign      foreach ($comment['replies'] as $rid){
433f0fda08aSwikidesign        $this->_print($rid, $data, $cid, $reply, $visible);
434f0fda08aSwikidesign      }
435f0fda08aSwikidesign      echo '</div>'.NL; // class="comment_replies"
436f0fda08aSwikidesign    }
437f0fda08aSwikidesign
438f0fda08aSwikidesign    if (!$comment['show']) echo '</div>'.NL; // class="comment_hidden"
439f0fda08aSwikidesign
440f0fda08aSwikidesign    // reply form
441f0fda08aSwikidesign    if ($reply == $cid){
442f0fda08aSwikidesign      echo '<div class="comment_replies">'.NL;
443f0fda08aSwikidesign      $this->_form('', 'add', $cid);
444f0fda08aSwikidesign      echo '</div>'.NL; // class="comment_replies"
445f0fda08aSwikidesign    }
446f0fda08aSwikidesign  }
447f0fda08aSwikidesign
448f0fda08aSwikidesign  /**
449f0fda08aSwikidesign   * Outputs the comment form
450f0fda08aSwikidesign   */
451f0fda08aSwikidesign  function _form($raw = '', $act = 'add', $cid = NULL){
452f0fda08aSwikidesign    global $lang;
453f0fda08aSwikidesign    global $conf;
454f0fda08aSwikidesign    global $ID;
455f0fda08aSwikidesign    global $INFO;
456*5ef1705fSiLoveiDo    global $ADMDISCUSSION;
457f0fda08aSwikidesign
458f0fda08aSwikidesign    // not for unregistered users when guest comments aren't allowed
459f0fda08aSwikidesign    if (!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) return false;
460f0fda08aSwikidesign
4611ba72c23Swikidesign    // fill $raw with $_REQUEST['text'] if it's empty (for failed CAPTCHA check)
4621ba72c23Swikidesign    if (!$raw && ($_REQUEST['comment'] == 'show')) $raw = $_REQUEST['text'];
463e7c760b3Swikidesign
464f0fda08aSwikidesign    ?>
465*5ef1705fSiLoveiDo
466*5ef1705fSiLoveiDo
467f0fda08aSwikidesign    <div class="comment_form">
468*5ef1705fSiLoveiDo      <form id="discussion__comment_form" method="post" action="<?php echo ((isset($ADMDISCUSSION['page']))?'':script()) ?>" accept-charset="<?php echo $lang['encoding'] ?>" onsubmit="return validate(this);">
469f0fda08aSwikidesign        <div class="no">
470*5ef1705fSiLoveiDo	<?php if (isset($ADMDISCUSSION['page'])) {
471*5ef1705fSiLoveiDo	?>
472*5ef1705fSiLoveiDo		<input type="hidden" name="page" value="<?php echo @$_GET['page'] ?>" />
473*5ef1705fSiLoveiDo	<?php
474*5ef1705fSiLoveiDo	}
475*5ef1705fSiLoveiDo	?>
476f0fda08aSwikidesign          <input type="hidden" name="id" value="<?php echo $ID ?>" />
477*5ef1705fSiLoveiDo          <input type="hidden" name="do" value="<?php echo ((isset($ADMDISCUSSION['page']))?'admin':'show') ?>" />
478f0fda08aSwikidesign          <input type="hidden" name="comment" value="<?php echo $act ?>" />
479f0fda08aSwikidesign    <?php
480f0fda08aSwikidesign
481f0fda08aSwikidesign    // for adding a comment
482f0fda08aSwikidesign    if ($act == 'add'){
483f0fda08aSwikidesign      ?>
484f0fda08aSwikidesign          <input type="hidden" name="reply" value="<?php echo $cid ?>" />
485f0fda08aSwikidesign      <?php
4861ba72c23Swikidesign      // for registered user (and we're not in admin import mode)
4871ba72c23Swikidesign      if ($conf['useacl'] && $_SERVER['REMOTE_USER']
4881ba72c23Swikidesign        && (!($this->getConf('adminimport') && ($INFO['perm'] == AUTH_ADMIN)))){
489f0fda08aSwikidesign      ?>
49065cdad39Swikidesign          <input type="hidden" name="user" value="<?php echo hsc($_SERVER['REMOTE_USER']) ?>" />
49165cdad39Swikidesign          <input type="hidden" name="name" value="<?php echo hsc($INFO['userinfo']['name']) ?>" />
49265cdad39Swikidesign          <input type="hidden" name="mail" value="<?php echo hsc($INFO['userinfo']['mail']) ?>" />
493f0fda08aSwikidesign      <?php
494f0fda08aSwikidesign      // for guest: show name and e-mail entry fields
495f0fda08aSwikidesign      } else {
496f0fda08aSwikidesign      ?>
497f0fda08aSwikidesign          <input type="hidden" name="user" value="<?php echo clientIP() ?>" />
498f0fda08aSwikidesign          <div class="comment_name">
499f0fda08aSwikidesign            <label class="block" for="discussion__comment_name">
500f0fda08aSwikidesign              <span><?php echo $lang['fullname'] ?>:</span>
501e7c760b3Swikidesign              <input type="text" class="edit" name="name" id="discussion__comment_name" size="50" tabindex="1" value="<?php echo hsc($_REQUEST['name'])?>" />
502f0fda08aSwikidesign            </label>
503f0fda08aSwikidesign          </div>
504f0fda08aSwikidesign          <div class="comment_mail">
505f0fda08aSwikidesign            <label class="block" for="discussion__comment_mail">
506f0fda08aSwikidesign              <span><?php echo $lang['email'] ?>:</span>
5071ba72c23Swikidesign              <input type="text" class="edit" name="mail" id="discussion__comment_mail" size="50" tabindex="2" value="<?php echo hsc($_REQUEST['mail'])?>" />
508f0fda08aSwikidesign            </label>
509f0fda08aSwikidesign          </div>
510f0fda08aSwikidesign      <?php
511f0fda08aSwikidesign      }
512f0fda08aSwikidesign
513f0fda08aSwikidesign      // allow entering an URL
514f0fda08aSwikidesign      if ($this->getConf('urlfield')){
515f0fda08aSwikidesign      ?>
516f0fda08aSwikidesign          <div class="comment_url">
517f0fda08aSwikidesign            <label class="block" for="discussion__comment_url">
518f0fda08aSwikidesign              <span><?php echo $this->getLang('url') ?>:</span>
519e7c760b3Swikidesign              <input type="text" class="edit" name="url" id="discussion__comment_url" size="50" tabindex="3" value="<?php echo hsc($_REQUEST['url'])?>" />
520f0fda08aSwikidesign            </label>
521f0fda08aSwikidesign          </div>
522f0fda08aSwikidesign      <?php
523f0fda08aSwikidesign      }
524f0fda08aSwikidesign
525f0fda08aSwikidesign      // allow entering an address
526f0fda08aSwikidesign      if ($this->getConf('addressfield')){
527f0fda08aSwikidesign      ?>
528f0fda08aSwikidesign          <div class="comment_address">
529f0fda08aSwikidesign            <label class="block" for="discussion__comment_address">
530f0fda08aSwikidesign              <span><?php echo $this->getLang('address') ?>:</span>
531e7c760b3Swikidesign              <input type="text" class="edit" name="address" id="discussion__comment_address" size="50" tabindex="4" value="<?php echo hsc($_REQUEST['address'])?>" />
532f0fda08aSwikidesign            </label>
533f0fda08aSwikidesign          </div>
534f0fda08aSwikidesign      <?php
535f0fda08aSwikidesign      }
536f0fda08aSwikidesign
537f0fda08aSwikidesign      // allow setting the comment date
5381ba72c23Swikidesign      if ($this->getConf('adminimport') && ($INFO['perm'] == AUTH_ADMIN)){
539f0fda08aSwikidesign      ?>
540f0fda08aSwikidesign          <div class="comment_date">
541f0fda08aSwikidesign            <label class="block" for="discussion__comment_date">
542f0fda08aSwikidesign              <span><?php echo $this->getLang('date') ?>:</span>
543f0fda08aSwikidesign              <input type="text" class="edit" name="date" id="discussion__comment_date" size="50" />
544f0fda08aSwikidesign            </label>
545f0fda08aSwikidesign          </div>
546f0fda08aSwikidesign      <?php
547f0fda08aSwikidesign      }
548f0fda08aSwikidesign
549f0fda08aSwikidesign    // for saving a comment
550f0fda08aSwikidesign    } else {
551f0fda08aSwikidesign    ?>
552f0fda08aSwikidesign          <input type="hidden" name="cid" value="<?php echo $cid ?>" />
553f0fda08aSwikidesign    <?php
554f0fda08aSwikidesign    }
555f0fda08aSwikidesign    ?>
556f0fda08aSwikidesign          <div class="comment_text">
55765cdad39Swikidesign            <textarea class="edit" name="text" cols="80" rows="10" id="discussion__comment_text" tabindex="5"><?php echo formText($raw) ?></textarea>
558f0fda08aSwikidesign          </div>
559e7c760b3Swikidesign    <?php //bad and dirty event insert hook
560e7c760b3Swikidesign    $evdata = array('writable' => true);
561e7c760b3Swikidesign    trigger_event('HTML_EDITFORM_INJECTION', $evdata);
562e7c760b3Swikidesign    ?>
563f0fda08aSwikidesign          <input class="button" type="submit" name="submit" value="<?php echo $lang['btn_save'] ?>" tabindex="6" />
564f0fda08aSwikidesign        </div>
565f0fda08aSwikidesign      </form>
566f0fda08aSwikidesign    </div>
567f0fda08aSwikidesign    <?php
568f0fda08aSwikidesign    if ($this->getConf('usecocomment')) echo $this->_coComment();
569f0fda08aSwikidesign  }
570f0fda08aSwikidesign
571f0fda08aSwikidesign  /**
572f0fda08aSwikidesign   * Adds a javascript to interact with coComments
573f0fda08aSwikidesign   */
574f0fda08aSwikidesign  function _coComment(){
575f0fda08aSwikidesign    global $ID;
576f0fda08aSwikidesign    global $conf;
577f0fda08aSwikidesign    global $INFO;
578*5ef1705fSiLoveiDo	 global  $ADMDISCUSSION;
579*5ef1705fSiLoveiDo	if (isset($ADMDISCUSSION['page'])) return;
580f0fda08aSwikidesign
581f0fda08aSwikidesign    $user = $_SERVER['REMOTE_USER'];
582f0fda08aSwikidesign
583f0fda08aSwikidesign    ?>
584f0fda08aSwikidesign    <script type="text/javascript"><!--//--><![CDATA[//><!--
585f0fda08aSwikidesign      var blogTool  = "DokuWiki";
586f0fda08aSwikidesign      var blogURL   = "<?php echo DOKU_URL ?>";
587f0fda08aSwikidesign      var blogTitle = "<?php echo $conf['title'] ?>";
588f0fda08aSwikidesign      var postURL   = "<?php echo wl($ID, '', true) ?>";
589f0fda08aSwikidesign      var postTitle = "<?php echo tpl_pagetitle($ID, true) ?>";
590f0fda08aSwikidesign    <?php
591f0fda08aSwikidesign    if ($user){
592f0fda08aSwikidesign    ?>
593f0fda08aSwikidesign      var commentAuthor = "<?php echo $INFO['userinfo']['name'] ?>";
594f0fda08aSwikidesign    <?php
595f0fda08aSwikidesign    } else {
596f0fda08aSwikidesign    ?>
597f0fda08aSwikidesign      var commentAuthorFieldName = "name";
598f0fda08aSwikidesign    <?php
599f0fda08aSwikidesign    }
600f0fda08aSwikidesign    ?>
601f0fda08aSwikidesign      var commentAuthorLoggedIn = <?php echo ($user ? 'true' : 'false') ?>;
602f0fda08aSwikidesign      var commentFormID         = "discussion__comment_form";
603f0fda08aSwikidesign      var commentTextFieldName  = "text";
604f0fda08aSwikidesign      var commentButtonName     = "submit";
605f0fda08aSwikidesign      var cocomment_force       = false;
606f0fda08aSwikidesign    //--><!]]></script>
607f0fda08aSwikidesign    <script type="text/javascript" src="http://www.cocomment.com/js/cocomment.js">
608f0fda08aSwikidesign    </script>
609f0fda08aSwikidesign    <?php
610f0fda08aSwikidesign  }
611f0fda08aSwikidesign
612f0fda08aSwikidesign  /**
613f0fda08aSwikidesign   * General button function
614f0fda08aSwikidesign   */
6151e46d176Swikidesign  function _button($cid, $label, $act, $jump = false){
616f0fda08aSwikidesign    global $ID;
617*5ef1705fSiLoveiDo    global $ADMDISCUSSION;
618*5ef1705fSiLoveiDo
6191e46d176Swikidesign    $anchor = ($jump ? '#discussion__comment_form' : '' );
620f0fda08aSwikidesign
621f0fda08aSwikidesign    ?>
622*5ef1705fSiLoveiDo    <form class="button" method="post" action="<?php echo ((isset($ADMDISCUSSION['page']))?'':script().$anchor) ?>">
623f0fda08aSwikidesign      <div class="no">
624*5ef1705fSiLoveiDo	<?php if (isset($ADMDISCUSSION['page'])) {
625*5ef1705fSiLoveiDo	?>
626*5ef1705fSiLoveiDo		<input type="hidden" name="page" value="<?php echo @$_GET['page'] ?>" />
627*5ef1705fSiLoveiDo	<?php
628*5ef1705fSiLoveiDo	}
629*5ef1705fSiLoveiDo	?>
630f0fda08aSwikidesign        <input type="hidden" name="id" value="<?php echo $ID ?>" />
631*5ef1705fSiLoveiDo        <input type="hidden" name="do" value="<?php echo ((isset($ADMDISCUSSION['page']))?'admin':'show') ?>" />
632f0fda08aSwikidesign        <input type="hidden" name="comment" value="<?php echo $act ?>" />
633f0fda08aSwikidesign        <input type="hidden" name="cid" value="<?php echo $cid ?>" />
634f0fda08aSwikidesign        <input type="submit" value="<?php echo $label ?>" class="button" title="<?php echo $label ?>" />
635f0fda08aSwikidesign      </div>
636f0fda08aSwikidesign    </form>
637f0fda08aSwikidesign    <?php
638f0fda08aSwikidesign    return true;
639f0fda08aSwikidesign  }
640f0fda08aSwikidesign
641f0fda08aSwikidesign  /**
642f0fda08aSwikidesign   * Adds an entry to the comments changelog
643f0fda08aSwikidesign   *
644f0fda08aSwikidesign   * @author Esther Brunner <wikidesign@gmail.com>
645f0fda08aSwikidesign   * @author Ben Coburn <btcoburn@silicodon.net>
646f0fda08aSwikidesign   */
647f0fda08aSwikidesign  function _addLogEntry($date, $id, $type = 'cc', $summary = '', $extra = ''){
648f0fda08aSwikidesign    global $conf;
649f0fda08aSwikidesign
650f0fda08aSwikidesign    $changelog = $conf['metadir'].'/_comments.changes';
651f0fda08aSwikidesign
652f0fda08aSwikidesign    if(!$date) $date = time(); //use current time if none supplied
653f0fda08aSwikidesign    $remote = $_SERVER['REMOTE_ADDR'];
654f0fda08aSwikidesign    $user   = $_SERVER['REMOTE_USER'];
655f0fda08aSwikidesign
656f0fda08aSwikidesign    $strip = array("\t", "\n");
657f0fda08aSwikidesign    $logline = array(
658f0fda08aSwikidesign      'date'  => $date,
659f0fda08aSwikidesign      'ip'    => $remote,
660f0fda08aSwikidesign      'type'  => str_replace($strip, '', $type),
661f0fda08aSwikidesign      'id'    => $id,
662f0fda08aSwikidesign      'user'  => $user,
663f0fda08aSwikidesign      'sum'   => str_replace($strip, '', $summary),
664f0fda08aSwikidesign      'extra' => str_replace($strip, '', $extra)
665f0fda08aSwikidesign    );
666f0fda08aSwikidesign
667f0fda08aSwikidesign    // add changelog line
668f0fda08aSwikidesign    $logline = implode("\t", $logline)."\n";
669f0fda08aSwikidesign    io_saveFile($changelog, $logline, true); //global changelog cache
670f0fda08aSwikidesign    $this->_trimRecentCommentsLog($changelog);
671f0fda08aSwikidesign  }
672f0fda08aSwikidesign
673f0fda08aSwikidesign  /**
674f0fda08aSwikidesign   * Trims the recent comments cache to the last $conf['changes_days'] recent
675f0fda08aSwikidesign   * changes or $conf['recent'] items, which ever is larger.
676f0fda08aSwikidesign   * The trimming is only done once a day.
677f0fda08aSwikidesign   *
678f0fda08aSwikidesign   * @author Ben Coburn <btcoburn@silicodon.net>
679f0fda08aSwikidesign   */
680f0fda08aSwikidesign  function _trimRecentCommentsLog($changelog){
681f0fda08aSwikidesign    global $conf;
682f0fda08aSwikidesign
683f0fda08aSwikidesign    if (@file_exists($changelog) &&
684f0fda08aSwikidesign      (filectime($changelog) + 86400) < time() &&
685f0fda08aSwikidesign      !@file_exists($changelog.'_tmp')){
686f0fda08aSwikidesign
687f0fda08aSwikidesign      io_lock($changelog);
688f0fda08aSwikidesign      $lines = file($changelog);
689f0fda08aSwikidesign      if (count($lines)<$conf['recent']) {
690f0fda08aSwikidesign          // nothing to trim
691f0fda08aSwikidesign          io_unlock($changelog);
692f0fda08aSwikidesign          return true;
693f0fda08aSwikidesign      }
694f0fda08aSwikidesign
695f0fda08aSwikidesign      io_saveFile($changelog.'_tmp', '');                  // presave tmp as 2nd lock
696f0fda08aSwikidesign      $trim_time = time() - $conf['recent_days']*86400;
697f0fda08aSwikidesign      $out_lines = array();
698f0fda08aSwikidesign
699f0fda08aSwikidesign      for ($i=0; $i<count($lines); $i++) {
700f0fda08aSwikidesign        $log = parseChangelogLine($lines[$i]);
701f0fda08aSwikidesign        if ($log === false) continue;                      // discard junk
702f0fda08aSwikidesign        if ($log['date'] < $trim_time) {
703f0fda08aSwikidesign          $old_lines[$log['date'].".$i"] = $lines[$i];     // keep old lines for now (append .$i to prevent key collisions)
704f0fda08aSwikidesign        } else {
705f0fda08aSwikidesign          $out_lines[$log['date'].".$i"] = $lines[$i];     // definitely keep these lines
706f0fda08aSwikidesign        }
707f0fda08aSwikidesign      }
708f0fda08aSwikidesign
709f0fda08aSwikidesign      // sort the final result, it shouldn't be necessary,
710f0fda08aSwikidesign      // however the extra robustness in making the changelog cache self-correcting is worth it
711f0fda08aSwikidesign      ksort($out_lines);
712f0fda08aSwikidesign      $extra = $conf['recent'] - count($out_lines);        // do we need extra lines do bring us up to minimum
713f0fda08aSwikidesign      if ($extra > 0) {
714f0fda08aSwikidesign        ksort($old_lines);
715f0fda08aSwikidesign        $out_lines = array_merge(array_slice($old_lines,-$extra),$out_lines);
716f0fda08aSwikidesign      }
717f0fda08aSwikidesign
718f0fda08aSwikidesign      // save trimmed changelog
719f0fda08aSwikidesign      io_saveFile($changelog.'_tmp', implode('', $out_lines));
720f0fda08aSwikidesign      @unlink($changelog);
721f0fda08aSwikidesign      if (!rename($changelog.'_tmp', $changelog)) {
722f0fda08aSwikidesign        // rename failed so try another way...
723f0fda08aSwikidesign        io_unlock($changelog);
724f0fda08aSwikidesign        io_saveFile($changelog, implode('', $out_lines));
725f0fda08aSwikidesign        @unlink($changelog.'_tmp');
726f0fda08aSwikidesign      } else {
727f0fda08aSwikidesign        io_unlock($changelog);
728f0fda08aSwikidesign      }
729f0fda08aSwikidesign      return true;
730f0fda08aSwikidesign    }
731f0fda08aSwikidesign  }
732f0fda08aSwikidesign
733f0fda08aSwikidesign  /**
734f0fda08aSwikidesign   * Sends a notify mail on new comment
735f0fda08aSwikidesign   *
736f0fda08aSwikidesign   * @param  array  $comment  data array of the new comment
737f0fda08aSwikidesign   *
738f0fda08aSwikidesign   * @author Andreas Gohr <andi@splitbrain.org>
739f0fda08aSwikidesign   * @author Esther Brunner <wikidesign@gmail.com>
740f0fda08aSwikidesign   */
741f0fda08aSwikidesign  function _notify($comment){
742f0fda08aSwikidesign    global $conf;
743f0fda08aSwikidesign    global $ID;
744f0fda08aSwikidesign
745cc0c9acdSwikidesign    if ((!$conf['subscribers']) && (!$conf['notify'])) return; //subscribers enabled?
746f0fda08aSwikidesign    $bcc  = subscriber_addresslist($ID);
747cc0c9acdSwikidesign    if ((empty($bcc)) && (!$conf['notify'])) return;
748cc0c9acdSwikidesign    $to   = $conf['notify'];
749f0fda08aSwikidesign    $text = io_readFile($this->localFN('subscribermail'));
750f0fda08aSwikidesign
7516046f25cSwikidesign    $search = array(
7526046f25cSwikidesign      '@PAGE@',
7536046f25cSwikidesign      '@TITLE@',
7546046f25cSwikidesign      '@DATE@',
7556046f25cSwikidesign      '@NAME@',
7566046f25cSwikidesign      '@TEXT@',
7576046f25cSwikidesign      '@UNSUBSCRIBE@',
7586046f25cSwikidesign      '@DOKUWIKIURL@',
7596046f25cSwikidesign    );
7606046f25cSwikidesign    $replace = array(
7616046f25cSwikidesign      $ID,
7626046f25cSwikidesign      $conf['title'],
7636046f25cSwikidesign      date($conf['dformat'], $comment['date']['created']),
7646046f25cSwikidesign      $comment['user']['name'],
7656046f25cSwikidesign      $comment['raw'],
7666046f25cSwikidesign      wl($ID, 'do=unsubscribe', true, '&'),
7676046f25cSwikidesign      DOKU_URL,
7686046f25cSwikidesign    );
7696046f25cSwikidesign    $text = str_replace($search, $replace, $text);
770f0fda08aSwikidesign
771f0fda08aSwikidesign    $subject = '['.$conf['title'].'] '.$this->getLang('mail_newcomment');
772f0fda08aSwikidesign
773f0fda08aSwikidesign    mail_send($to, $subject, $text, $conf['mailfrom'], '', $bcc);
774f0fda08aSwikidesign  }
775f0fda08aSwikidesign
776f0fda08aSwikidesign  /**
777f0fda08aSwikidesign   * Counts the number of visible comments
778f0fda08aSwikidesign   */
779f0fda08aSwikidesign  function _count($data){
780f0fda08aSwikidesign    $number = 0;
781f0fda08aSwikidesign    foreach ($data['comments'] as $cid => $comment){
782f0fda08aSwikidesign      if ($comment['parent']) continue;
783f0fda08aSwikidesign      if (!$comment['show']) continue;
784f0fda08aSwikidesign      $number++;
785f0fda08aSwikidesign      $rids = $comment['replies'];
786f0fda08aSwikidesign      if (count($rids)) $number = $number + $this->_countReplies($data, $rids);
787f0fda08aSwikidesign    }
788f0fda08aSwikidesign    return $number;
789f0fda08aSwikidesign  }
790f0fda08aSwikidesign
791f0fda08aSwikidesign  function _countReplies(&$data, $rids){
792f0fda08aSwikidesign    $number = 0;
793f0fda08aSwikidesign    foreach ($rids as $rid){
7942ee3dca3Swikidesign      if (!isset($data['comments'][$rid])) continue; // reply was removed
795f0fda08aSwikidesign      if (!$data['comments'][$rid]['show']) continue;
796f0fda08aSwikidesign      $number++;
797f0fda08aSwikidesign      $rids = $data['comments'][$rid]['replies'];
798f0fda08aSwikidesign      if (count($rids)) $number = $number + $this->_countReplies($data, $rids);
799f0fda08aSwikidesign    }
800f0fda08aSwikidesign    return $number;
801f0fda08aSwikidesign  }
802f0fda08aSwikidesign
803f0fda08aSwikidesign  /**
804f0fda08aSwikidesign   * Renders the comment text
805f0fda08aSwikidesign   */
806f0fda08aSwikidesign  function _render($raw){
807f0fda08aSwikidesign    if ($this->getConf('wikisyntaxok')){
808f0fda08aSwikidesign      $xhtml = $this->render($raw);
809f0fda08aSwikidesign    } else { // wiki syntax not allowed -> just encode special chars
810f0fda08aSwikidesign      $xhtml = htmlspecialchars(trim($raw));
811f0fda08aSwikidesign    }
812f0fda08aSwikidesign    return $xhtml;
813f0fda08aSwikidesign  }
814f0fda08aSwikidesign
815f0fda08aSwikidesign  /**
816479dd10fSwikidesign   * Adds a TOC item for the discussion section
817479dd10fSwikidesign   */
818479dd10fSwikidesign  function add_toc_item(&$event, $param){
819479dd10fSwikidesign    if ($event->data[0] != 'xhtml') return; // nothing to do for us
820479dd10fSwikidesign    if (!$this->_hasDiscussion()) return;   // no discussion section
821479dd10fSwikidesign
822479dd10fSwikidesign    $pattern = '/<div id="toc__inside">(.*?)<\/div>\s<\/div>/s';
823479dd10fSwikidesign    if (!preg_match($pattern, $event->data[1], $match)) return; // no TOC on this page
824479dd10fSwikidesign
825479dd10fSwikidesign    // ok, then let's do it!
826479dd10fSwikidesign    global $conf;
827479dd10fSwikidesign
828479dd10fSwikidesign    $title   = $this->getLang('discussion');
829479dd10fSwikidesign    $section = '#discussion__section';
830479dd10fSwikidesign    $level   = 3 - $conf['toptoclevel'];
831479dd10fSwikidesign
832479dd10fSwikidesign    $item = '<li class="level'.$level.'"><div class="li"><span class="li"><a href="'.
833479dd10fSwikidesign      $section.'" class="toc">'.$title.'</a></span></div></li>';
834479dd10fSwikidesign
835479dd10fSwikidesign    if ($level == 1) $search = "</ul>\n</div>";
836479dd10fSwikidesign    else $search = "</ul>\n</li></ul>\n</div>";
837479dd10fSwikidesign
838479dd10fSwikidesign    $new = str_replace($search, $item.$search, $match[0]);
839479dd10fSwikidesign    $event->data[1] = preg_replace($pattern, $new, $event->data[1]);
840479dd10fSwikidesign  }
841479dd10fSwikidesign
842479dd10fSwikidesign  /**
843479dd10fSwikidesign   * Finds out whether there is a discussion section for the current page
844479dd10fSwikidesign   */
845479dd10fSwikidesign  function _hasDiscussion(){
846479dd10fSwikidesign    global $ID;
847479dd10fSwikidesign
848479dd10fSwikidesign    $cfile = metaFN($ID, '.comments');
849479dd10fSwikidesign
850479dd10fSwikidesign    if (!@file_exists($cfile)){
851479dd10fSwikidesign      if ($this->getConf('automatic')) return true;
852479dd10fSwikidesign      else return false;
853479dd10fSwikidesign    }
854479dd10fSwikidesign
855479dd10fSwikidesign    $comments = unserialize(io_readFile($cfile, false));
856479dd10fSwikidesign
857479dd10fSwikidesign    $num = $comments['number'];
858479dd10fSwikidesign    if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return false;
859479dd10fSwikidesign    else return true;
860479dd10fSwikidesign  }
861479dd10fSwikidesign
862479dd10fSwikidesign  /**
863e7c760b3Swikidesign   * Checks if 'newthread' was given as action or the comment form was submitted
864f0fda08aSwikidesign   */
865f0fda08aSwikidesign  function handle_act_preprocess(&$event, $param){
866e7c760b3Swikidesign    if ($event->data == 'newthread'){
8672e80cd5fSwikidesign      // we can handle it -> prevent others
8682e80cd5fSwikidesign      // $event->stopPropagation();
8692e80cd5fSwikidesign      $event->preventDefault();
8702e80cd5fSwikidesign
8716046f25cSwikidesign      $event->data = $this->_newThread();
872e7c760b3Swikidesign    }
873e7c760b3Swikidesign    if ((in_array($_REQUEST['comment'], array('add', 'save')))
874e7c760b3Swikidesign      && (@file_exists(DOKU_PLUGIN.'captcha/action.php'))){
8756046f25cSwikidesign      $this->_captchaCheck();
876e7c760b3Swikidesign    }
877e7c760b3Swikidesign  }
878f0fda08aSwikidesign
879e7c760b3Swikidesign  /**
880e7c760b3Swikidesign   * Creates a new thread page
881e7c760b3Swikidesign   */
8826046f25cSwikidesign  function _newThread(){
883f0fda08aSwikidesign    global $ID;
8842e80cd5fSwikidesign    global $INFO;
885f0fda08aSwikidesign
8861ea794e5Swikidesign    $ns    = cleanID($_REQUEST['ns']);
887f0fda08aSwikidesign    $title = str_replace(':', '', $_REQUEST['title']);
8882e80cd5fSwikidesign    $back  = $ID;
8892e80cd5fSwikidesign    $ID    = ($ns ? $ns.':' : '').cleanID($title);
8902e80cd5fSwikidesign    $INFO  = pageinfo();
891f0fda08aSwikidesign
892f0fda08aSwikidesign    // check if we are allowed to create this file
8932e80cd5fSwikidesign    if ($INFO['perm'] >= AUTH_CREATE){
894f0fda08aSwikidesign
895f0fda08aSwikidesign      //check if locked by anyone - if not lock for my self
8962e80cd5fSwikidesign      if ($INFO['locked']) return 'locked';
8972e80cd5fSwikidesign      else lock($ID);
898f0fda08aSwikidesign
899f0fda08aSwikidesign      // prepare the new thread file with default stuff
9002e80cd5fSwikidesign      if (!@file_exists($INFO['filepath'])){
901f0fda08aSwikidesign        global $TEXT;
902f0fda08aSwikidesign        global $conf;
903f0fda08aSwikidesign
9042e80cd5fSwikidesign        $TEXT = pageTemplate(array(($ns ? $ns.':' : '').$title));
9051433886fSwikidesign        if (!$TEXT){
9061433886fSwikidesign          $TEXT = "<- [[:$back]]\n\n====== $title ======\n\n";
9071433886fSwikidesign          if ($this->getConf('usegravatar'))
9081433886fSwikidesign            $TEXT .= '{{gravatar>'.$INFO['userinfo']['mail'].' }} ';
9098820dcabSwikidesign          $TEXT .= "//".$INFO['userinfo']['name'].", ".date($conf['dformat']).": //\n\n";
910f844c865Swikidesign          if ((@file_exists(DOKU_PLUGIN.'tag/syntax/tag.php'))
911f844c865Swikidesign            && (!plugin_isdisabled('tag')))
9128820dcabSwikidesign            $TEXT .= "\n\n{{tag>}}";
9138820dcabSwikidesign          $TEXT .= "\n\n~~DISCUSSION~~";
9142e80cd5fSwikidesign        }
9152e80cd5fSwikidesign        return 'preview';
916f0fda08aSwikidesign      } else {
9172e80cd5fSwikidesign        return 'edit';
918f0fda08aSwikidesign      }
919f0fda08aSwikidesign    } else {
9202e80cd5fSwikidesign      return 'show';
921f0fda08aSwikidesign    }
922f0fda08aSwikidesign  }
923f0fda08aSwikidesign
924e7c760b3Swikidesign  /**
925e7c760b3Swikidesign   * Checks if the CAPTCHA string submitted is valid
926e7c760b3Swikidesign   *
927e7c760b3Swikidesign   * @author     Andreas Gohr <gohr@cosmocode.de>
928e7c760b3Swikidesign   * @adaption   Esther Brunner <wikidesign@gmail.com>
929e7c760b3Swikidesign   */
9306046f25cSwikidesign  function _captchaCheck(){
931e7c760b3Swikidesign    if (@file_exists(DOKU_PLUGIN.'captcha/disabled')) return; // CAPTCHA is disabled
932e7c760b3Swikidesign
933e7c760b3Swikidesign    require_once(DOKU_PLUGIN.'captcha/action.php');
934e7c760b3Swikidesign    $captcha = new action_plugin_captcha;
935e7c760b3Swikidesign
936d1c29589Swikidesign    // do nothing if logged in user and no CAPTCHA required
937d1c29589Swikidesign    if (!$captcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return;
938d1c29589Swikidesign
939e7c760b3Swikidesign    // compare provided string with decrypted captcha
940e7c760b3Swikidesign    $rand = PMA_blowfish_decrypt($_REQUEST['plugin__captcha_secret'], auth_cookiesalt());
941e7c760b3Swikidesign    $code = $captcha->_generateCAPTCHA($captcha->_fixedIdent(), $rand);
942e7c760b3Swikidesign
943e7c760b3Swikidesign    if (!$_REQUEST['plugin__captcha_secret'] ||
944e7c760b3Swikidesign      !$_REQUEST['plugin__captcha'] ||
945e7c760b3Swikidesign      strtoupper($_REQUEST['plugin__captcha']) != $code){
946e7c760b3Swikidesign
947e7c760b3Swikidesign      // CAPTCHA test failed! Continue to edit instead of saving
948e7c760b3Swikidesign      msg($captcha->getLang('testfailed'), -1);
949e7c760b3Swikidesign      if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit';
950e7c760b3Swikidesign      elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show';
951e7c760b3Swikidesign    }
952e7c760b3Swikidesign    // if we arrive here it was a valid save
953e7c760b3Swikidesign  }
954e7c760b3Swikidesign
955f0fda08aSwikidesign}
956f0fda08aSwikidesign
957f0fda08aSwikidesign//Setup VIM: ex: et ts=4 enc=utf-8 :
958