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 ); 69b8fdc796SMichael Klier $contr->register_hook( 70b8fdc796SMichael Klier 'AJAX_CALL_UNKNOWN', 71b8fdc796SMichael Klier 'BEFORE', 72b8fdc796SMichael Klier $this, 73b8fdc796SMichael Klier 'handle_ajax_call', 74b8fdc796SMichael Klier array() 75b8fdc796SMichael Klier ); 765886c85bSMichael Klier $contr->register_hook( 775886c85bSMichael Klier 'TPL_TOC_RENDER', 785886c85bSMichael Klier 'BEFORE', 795886c85bSMichael Klier $this, 805886c85bSMichael Klier 'handle_toc_render', 815886c85bSMichael Klier array() 825886c85bSMichael Klier ); 83b8fdc796SMichael Klier } 84b8fdc796SMichael Klier 85b8fdc796SMichael Klier /** 86b8fdc796SMichael Klier * Preview Comments 87b8fdc796SMichael Klier * 88b8fdc796SMichael Klier * @author Michael Klier <chi@chimeric.de> 89b8fdc796SMichael Klier */ 90b8fdc796SMichael Klier function handle_ajax_call(&$event, $params) { 91b8fdc796SMichael Klier if($event->data != 'discussion_preview') return; 92b8fdc796SMichael Klier $event->preventDefault(); 93b8fdc796SMichael Klier $event->stopPropagation(); 94b8fdc796SMichael Klier print p_locale_xhtml('preview'); 95b8fdc796SMichael Klier print '<div class="comment_preview">'; 96b8fdc796SMichael Klier if(!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) { 97b8fdc796SMichael Klier print p_locale_xhtml('denied'); 98b8fdc796SMichael Klier } else { 9987bb4e97SMichael Klier print $this->_render($_REQUEST['comment']); 100b8fdc796SMichael Klier } 101b8fdc796SMichael Klier print '</div>'; 1022d4bee9aSMichael Klier } 1032d4bee9aSMichael Klier 1042d4bee9aSMichael Klier /** 1055886c85bSMichael Klier * Adds a TOC item if a discussion exists 1065886c85bSMichael Klier * 1075886c85bSMichael Klier * @author Michael Klier <chi@chimeric.de> 1085886c85bSMichael Klier */ 1095886c85bSMichael Klier function handle_toc_render(&$event, $params) { 1105886c85bSMichael Klier global $ID; 1115886c85bSMichael Klier if($this->_hasDiscussion($title) && $event->data) { 1125886c85bSMichael Klier $tocitem = array( 'hid' => 'discussion__section', 1135886c85bSMichael Klier 'title' => $this->getLang('discussion'), 1145886c85bSMichael Klier 'type' => 'ul', 1155886c85bSMichael Klier 'level' => 1 ); 1165886c85bSMichael Klier 1175886c85bSMichael Klier array_push($event->data, $tocitem); 1185886c85bSMichael Klier } 1195886c85bSMichael Klier } 1205886c85bSMichael Klier 1215886c85bSMichael Klier /** 1222d4bee9aSMichael Klier * Modify Tollbar for use with discussion plugin 1232d4bee9aSMichael Klier * 1242d4bee9aSMichael Klier * @author Michael Klier <chi@chimeric.de> 1252d4bee9aSMichael Klier */ 1262d4bee9aSMichael Klier function handle_toolbar_define(&$event, $param) { 1272d4bee9aSMichael Klier global $ACT; 1282d4bee9aSMichael Klier if($ACT != 'show') return; 1292d4bee9aSMichael Klier 1301de52da1SMichael Klier if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) { 1312d4bee9aSMichael Klier $toolbar = array(); 1322d4bee9aSMichael Klier foreach($event->data as $btn) { 1332d4bee9aSMichael Klier if($btn['type'] == 'mediapopup') continue; 1342d4bee9aSMichael Klier if($btn['type'] == 'signature') continue; 1352d4bee9aSMichael Klier if(preg_match("/=+?/", $btn['open'])) continue; 1362d4bee9aSMichael Klier array_push($toolbar, $btn); 1372d4bee9aSMichael Klier } 1382d4bee9aSMichael Klier $event->data = $toolbar; 1392d4bee9aSMichael Klier } 1402d4bee9aSMichael Klier } 1412d4bee9aSMichael Klier 1422d4bee9aSMichael Klier /** 1432d4bee9aSMichael Klier * Dirty workaround to add a toolbar to the discussion plugin 1442d4bee9aSMichael Klier * 1452d4bee9aSMichael Klier * @author Michael Klier <chi@chimeric.de> 1462d4bee9aSMichael Klier */ 1472d4bee9aSMichael Klier function handle_tpl_metaheader_output(&$event, $param) { 1482d4bee9aSMichael Klier global $ACT; 1492d4bee9aSMichael Klier global $ID; 1502d4bee9aSMichael Klier if($ACT != 'show') return; 1512d4bee9aSMichael Klier 1522d4bee9aSMichael Klier // FIXME check if this works for global discussion/on too 1531de52da1SMichael Klier if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) { 1542d4bee9aSMichael Klier // FIXME ugly workaround, replace this once DW the toolbar code is more flexible 1552d4bee9aSMichael Klier array_unshift($event->data['script'], array('type' => 'text/javascript', 'charset' => 'utf-8', '_data' => '', 'src' => DOKU_BASE.'lib/scripts/edit.js')); 1562d4bee9aSMichael Klier @require_once(DOKU_INC.'inc/toolbar.php'); 1572d4bee9aSMichael Klier ob_start(); 1582d4bee9aSMichael Klier print 'NS = "' . getNS($ID) . '";'; // we have to define NS, otherwise we get get JS errors 1592d4bee9aSMichael Klier toolbar_JSdefines('toolbar'); 1602d4bee9aSMichael Klier $script = ob_get_clean(); 1612d4bee9aSMichael Klier array_push($event->data['script'], array('type' => 'text/javascript', 'charset' => "utf-8", '_data' => $script)); 1622d4bee9aSMichael Klier } 163f0fda08aSwikidesign } 164f0fda08aSwikidesign 165f0fda08aSwikidesign /** 166a1d93126SGina Haeussge * Handles comment actions, dispatches data processing routines 167f0fda08aSwikidesign */ 168a1d93126SGina Haeussge function handle_act_preprocess(&$event, $param) { 169284f9aa2SMichael Klier global $ID; 170d7eb2e24SMichael Klier global $INFO; 171284f9aa2SMichael Klier global $conf; 1723011fb8bSMichael Klier global $lang; 173573e23a1Swikidesign 174a1d93126SGina Haeussge // handle newthread ACTs 175a1d93126SGina Haeussge if ($event->data == 'newthread') { 176a1d93126SGina Haeussge // we can handle it -> prevent others 177a1d93126SGina Haeussge $event->preventDefault(); 178a1d93126SGina Haeussge $event->data = $this->_newThread(); 179a1d93126SGina Haeussge } 180a1d93126SGina Haeussge 181a1d93126SGina Haeussge // enable captchas 182*bd6dc08eSAdrian Schlegel if (in_array($_REQUEST['comment'], array('add', 'save'))) { 183*bd6dc08eSAdrian Schlegel if (@file_exists(DOKU_PLUGIN.'captcha/action.php')) { 184a1d93126SGina Haeussge $this->_captchaCheck(); 185a1d93126SGina Haeussge } 186*bd6dc08eSAdrian Schlegel if (@file_exists(DOKU_PLUGIN.'recaptcha/action.php')) { 187*bd6dc08eSAdrian Schlegel $this->_recaptchaCheck(); 188*bd6dc08eSAdrian Schlegel } 189*bd6dc08eSAdrian Schlegel } 190a1d93126SGina Haeussge 1913011fb8bSMichael Klier // if we are not in show mode or someone wants to unsubscribe, that was all for now 192b2f2e866SMichael Klier if ($event->data != 'show' && $event->data != 'discussion_unsubscribe' && $event->data != 'discussion_confirmsubscribe') return; 193a1d93126SGina Haeussge 194b2f2e866SMichael Klier if ($event->data == 'discussion_unsubscribe' or $event->data == 'discussion_confirmsubscribe') { 1959881d835SMichael Klier // ok we can handle it prevent others 1969881d835SMichael Klier $event->preventDefault(); 1979881d835SMichael Klier 1983011fb8bSMichael Klier if (!isset($_REQUEST['hash'])) { 1999881d835SMichael Klier return false; 2003011fb8bSMichael Klier } else { 2013011fb8bSMichael Klier $file = metaFN($ID, '.comments'); 2023011fb8bSMichael Klier $data = unserialize(io_readFile($file)); 2030c54624fSGina Haeussge $themail = ''; 2049881d835SMichael Klier foreach($data['subscribers'] as $mail => $info) { 2059881d835SMichael Klier // convert old style subscribers just in case 2069881d835SMichael Klier if(!is_array($info)) { 2079881d835SMichael Klier $hash = $data['subscribers'][$mail]; 2089881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 2099881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 2109881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 2119881d835SMichael Klier } 2129881d835SMichael Klier 2139881d835SMichael Klier if ($data['subscribers'][$mail]['hash'] == $_REQUEST['hash']) { 2140c54624fSGina Haeussge $themail = $mail; 2150c54624fSGina Haeussge } 2160c54624fSGina Haeussge } 2170c54624fSGina Haeussge 2180c54624fSGina Haeussge if($themail != '') { 219b2f2e866SMichael Klier if($event->data == 'discussion_unsubscribe') { 2200c54624fSGina Haeussge unset($data['subscribers'][$themail]); 2210c54624fSGina Haeussge msg(sprintf($lang['unsubscribe_success'], $themail, $ID), 1); 222b2f2e866SMichael Klier } elseif($event->data == 'discussion_confirmsubscribe') { 2230c54624fSGina Haeussge $data['subscribers'][$themail]['active'] = true; 2240c54624fSGina Haeussge msg(sprintf($lang['subscribe_success'], $themail, $ID), 1); 2259881d835SMichael Klier } 2269881d835SMichael Klier io_saveFile($file, serialize($data)); 2273011fb8bSMichael Klier $event->data = 'show'; 2283011fb8bSMichael Klier return true; 2299881d835SMichael Klier } else { 2303011fb8bSMichael Klier return false; 2313011fb8bSMichael Klier } 2329881d835SMichael Klier } 2333011fb8bSMichael Klier } else { 234a1d93126SGina Haeussge // do the data processing for comments 235f0fda08aSwikidesign $cid = $_REQUEST['cid']; 236f0fda08aSwikidesign switch ($_REQUEST['comment']) { 237f0fda08aSwikidesign case 'add': 23837e3c825SMichael Klier if(empty($_REQUEST['text'])) return; // don't add empty comments 23994c5d164SMichael Klier if(isset($_SERVER['REMOTE_USER']) && !$this->getConf('adminimport')) { 24094c5d164SMichael Klier $comment['user']['id'] = $_SERVER['REMOTE_USER']; 24194c5d164SMichael Klier $comment['user']['name'] = $INFO['userinfo']['name']; 24294c5d164SMichael Klier $comment['user']['mail'] = $INFO['userinfo']['mail']; 24394c5d164SMichael Klier } elseif((isset($_SERVER['REMOTE_USER']) && $this->getConf('adminimport') && auth_ismanager()) || !isset($_SERVER['REMOTE_USER'])) { 24437e3c825SMichael Klier if(empty($_REQUEST['name']) or empty($_REQUEST['mail'])) return // don't add anonymous comments 24594c5d164SMichael Klier $comment['user']['id'] = 'test'.hsc($_REQUEST['user']); 24694c5d164SMichael Klier $comment['user']['name'] = hsc($_REQUEST['name']); 24794c5d164SMichael Klier $comment['user']['mail'] = hsc($_REQUEST['mail']); 24894c5d164SMichael Klier } 24994c5d164SMichael Klier $comment['user']['address'] = ($this->getConf('addressfield')) ? hsc($_REQUEST['address']) : ''; 250a10b5c98SMichael Klier $comment['user']['url'] = ($this->getConf('urlfield')) ? $this->_checkURL($_REQUEST['url']) : ''; 25194c5d164SMichael Klier $comment['subscribe'] = ($this->getConf('subscribe')) ? $_REQUEST['subscribe'] : ''; 25294c5d164SMichael Klier $comment['date'] = array('created' => $_REQUEST['date']); 25394c5d164SMichael Klier $comment['raw'] = cleanText($_REQUEST['text']); 254f0fda08aSwikidesign $repl = $_REQUEST['reply']; 255b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 256a44bc9f7SMichael Klier $comment['show'] = false; 257a44bc9f7SMichael Klier } else { 258a44bc9f7SMichael Klier $comment['show'] = true; 259a44bc9f7SMichael Klier } 260f0fda08aSwikidesign $this->_add($comment, $repl); 261f0fda08aSwikidesign break; 262f0fda08aSwikidesign 263f0fda08aSwikidesign case 'save': 264f0fda08aSwikidesign $raw = cleanText($_REQUEST['text']); 265264b7327Swikidesign $this->_save(array($cid), $raw); 266f0fda08aSwikidesign break; 267f0fda08aSwikidesign 2681e46d176Swikidesign case 'delete': 269264b7327Swikidesign $this->_save(array($cid), ''); 2702ee3dca3Swikidesign break; 2711e46d176Swikidesign 272f0fda08aSwikidesign case 'toogle': 273264b7327Swikidesign $this->_save(array($cid), '', 'toogle'); 274f0fda08aSwikidesign break; 275a1d93126SGina Haeussge } 2763011fb8bSMichael Klier } 277a1d93126SGina Haeussge } 278a1d93126SGina Haeussge 279a1d93126SGina Haeussge /** 280a1d93126SGina Haeussge * Main function; dispatches the visual comment actions 281a1d93126SGina Haeussge */ 282a1d93126SGina Haeussge function comments(&$event, $param) { 283a1d93126SGina Haeussge if ($event->data != 'show') return; // nothing to do for us 284a1d93126SGina Haeussge 285a1d93126SGina Haeussge $cid = $_REQUEST['cid']; 286a1d93126SGina Haeussge switch ($_REQUEST['comment']) { 287a1d93126SGina Haeussge case 'edit': 288a1d93126SGina Haeussge $this->_show(NULL, $cid); 289a1d93126SGina Haeussge break; 2902b18adb9SMichael Klier default: 291f0fda08aSwikidesign $this->_show($cid); 2922b18adb9SMichael Klier break; 293f0fda08aSwikidesign } 294f0fda08aSwikidesign } 295f0fda08aSwikidesign 296f0fda08aSwikidesign /** 297a1d93126SGina Haeussge * Redirects browser to given comment anchor 298a1d93126SGina Haeussge */ 299a1d93126SGina Haeussge function _redirect($cid) { 300a1d93126SGina Haeussge global $ID; 301a1d93126SGina Haeussge global $ACT; 302a1d93126SGina Haeussge 303a1d93126SGina Haeussge if ($ACT !== 'show') return; 304a44bc9f7SMichael Klier 305b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 306a44bc9f7SMichael Klier msg($this->getLang('moderation'), 1); 307a44bc9f7SMichael Klier @session_start(); 308a44bc9f7SMichael Klier global $MSG; 309a44bc9f7SMichael Klier $_SESSION[DOKU_COOKIE]['msg'] = $MSG; 310a44bc9f7SMichael Klier session_write_close(); 311a44bc9f7SMichael Klier $url = wl($ID); 312a44bc9f7SMichael Klier } else { 313a44bc9f7SMichael Klier $url = wl($ID) . '#comment_' . $cid; 314a44bc9f7SMichael Klier } 315a44bc9f7SMichael Klier send_redirect($url); 3166d1f4f20SGina Haeussge exit(); 317a1d93126SGina Haeussge } 318a1d93126SGina Haeussge 319a1d93126SGina Haeussge /** 320f0fda08aSwikidesign * Shows all comments of the current page 321f0fda08aSwikidesign */ 322f0fda08aSwikidesign function _show($reply = NULL, $edit = NULL) { 3232b18adb9SMichael Klier global $ID; 3242b18adb9SMichael Klier global $INFO; 3252b18adb9SMichael Klier global $ACT; 326573e23a1Swikidesign 327479dd10fSwikidesign // get .comments meta file name 328f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 329f0fda08aSwikidesign 3304aeddd8eSMichael Klier if (!$INFO['exists']) return; 3312b18adb9SMichael Klier if (!@file_exists($file) && !$this->getConf('automatic')) return false; 3327d091ad1SMichael Klier if (!$_SERVER['REMOTE_USER'] && !$this->getConf('showguests')) return false; 333f0fda08aSwikidesign 3342b18adb9SMichael Klier // load data 3352b18adb9SMichael Klier if (@file_exists($file)) { 3362b18adb9SMichael Klier $data = unserialize(io_readFile($file, false)); 337479dd10fSwikidesign if (!$data['status']) return false; // comments are turned off 3382b18adb9SMichael Klier } elseif (!@file_exists($file) && $this->getConf('automatic') && $INFO['exists']) { 3392b18adb9SMichael Klier // set status to show the comment form 3402b18adb9SMichael Klier $data['status'] = 1; 3412b18adb9SMichael Klier $data['number'] = 0; 3422b18adb9SMichael Klier } 343f0fda08aSwikidesign 344a1599850SMichael Klier // show discussion wrapper only on certain circumstances 345a1599850SMichael Klier $cnt = count($data['comments']); 34664018b99SMichael Klier $keys = @array_keys($data['comments']); 347a1599850SMichael Klier if($cnt > 1 || ($cnt == 1 && $data['comments'][$keys[0]]['show'] == 1) || $this->getConf('allowguests') || isset($_SERVER['REMOTE_USER'])) { 348a1599850SMichael Klier $show = true; 349f0fda08aSwikidesign // section title 35007c376bbSwikidesign $title = ($data['title'] ? hsc($data['title']) : $this->getLang('discussion')); 3514a0a1bd2Swikidesign ptln('<div class="comment_wrapper">'); 3524a0a1bd2Swikidesign ptln('<h2><a name="discussion__section" id="discussion__section">', 2); 3534a0a1bd2Swikidesign ptln($title, 4); 3544a0a1bd2Swikidesign ptln('</a></h2>', 2); 3554a0a1bd2Swikidesign ptln('<div class="level2 hfeed">', 2); 356a1599850SMichael Klier } 357a1599850SMichael Klier 358f0fda08aSwikidesign // now display the comments 359f0fda08aSwikidesign if (isset($data['comments'])) { 36031aab30eSGina Haeussge if (!$this->getConf('usethreading')) { 36131aab30eSGina Haeussge $data['comments'] = $this->_flattenThreads($data['comments']); 36231aab30eSGina Haeussge uasort($data['comments'], '_sortCallBack'); 36331aab30eSGina Haeussge } 364dbd9d5cdSMichael Klier if($this->getConf('newestfirst')) { 365dbd9d5cdSMichael Klier $data['comments'] = array_reverse($data['comments']); 366dbd9d5cdSMichael Klier } 367f0fda08aSwikidesign foreach ($data['comments'] as $key => $value) { 368f0fda08aSwikidesign if ($key == $edit) $this->_form($value['raw'], 'save', $edit); // edit form 369f0fda08aSwikidesign else $this->_print($key, $data, '', $reply); 370f0fda08aSwikidesign } 371f0fda08aSwikidesign } 372f0fda08aSwikidesign 373f0fda08aSwikidesign // comment form 37431aab30eSGina Haeussge if (($data['status'] == 1) && (!$reply || !$this->getConf('usethreading')) && !$edit) $this->_form(''); 375f0fda08aSwikidesign 376a1599850SMichael Klier if($show) { 3774a0a1bd2Swikidesign ptln('</div>', 2); // level2 hfeed 3784a0a1bd2Swikidesign ptln('</div>'); // comment_wrapper 379a1599850SMichael Klier } 380f0fda08aSwikidesign 381f0fda08aSwikidesign return true; 382f0fda08aSwikidesign } 383f0fda08aSwikidesign 38431aab30eSGina Haeussge function _flattenThreads($comments, $keys = null) { 38531aab30eSGina Haeussge if (is_null($keys)) 38631aab30eSGina Haeussge $keys = array_keys($comments); 38731aab30eSGina Haeussge 38831aab30eSGina Haeussge foreach($keys as $cid) { 38931aab30eSGina Haeussge if (!empty($comments[$cid]['replies'])) { 39031aab30eSGina Haeussge $rids = $comments[$cid]['replies']; 39131aab30eSGina Haeussge $comments = $this->_flattenThreads($comments, $rids); 39231aab30eSGina Haeussge $comments[$cid]['replies'] = array(); 39331aab30eSGina Haeussge } 39431aab30eSGina Haeussge $comments[$cid]['parent'] = ''; 39531aab30eSGina Haeussge } 39631aab30eSGina Haeussge return $comments; 39731aab30eSGina Haeussge } 39831aab30eSGina Haeussge 399f0fda08aSwikidesign /** 400f0fda08aSwikidesign * Adds a new comment and then displays all comments 401f0fda08aSwikidesign */ 402f0fda08aSwikidesign function _add($comment, $parent) { 4033011fb8bSMichael Klier global $lang; 4043011fb8bSMichael Klier global $ID; 4053011fb8bSMichael Klier global $TEXT; 406f0fda08aSwikidesign 407f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 408f0fda08aSwikidesign $TEXT = $comment['raw']; 409f0fda08aSwikidesign 410f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 411f0fda08aSwikidesign if (checkwordblock()) { 412f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 413f0fda08aSwikidesign return false; 414f0fda08aSwikidesign } 415f0fda08aSwikidesign 4163011fb8bSMichael Klier if ((!$this->getConf('allowguests')) 4173011fb8bSMichael Klier && ($comment['user']['id'] != $_SERVER['REMOTE_USER'])) 4183011fb8bSMichael Klier return false; // guest comments not allowed 4193011fb8bSMichael Klier 420f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 421f0fda08aSwikidesign 422f0fda08aSwikidesign // get discussion meta file name 423f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 424f0fda08aSwikidesign 4252b18adb9SMichael Klier // create comments file if it doesn't exist yet 4262b18adb9SMichael Klier if(!@file_exists($file)) { 4272b18adb9SMichael Klier $data = array('status' => 1, 'number' => 0); 4282b18adb9SMichael Klier io_saveFile($file, serialize($data)); 4292b18adb9SMichael Klier } else { 430f0fda08aSwikidesign $data = array(); 431f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 432f0fda08aSwikidesign if ($data['status'] != 1) return false; // comments off or closed 4332b18adb9SMichael Klier } 4342b18adb9SMichael Klier 4353011fb8bSMichael Klier if ($comment['date']['created']) { 4363011fb8bSMichael Klier $date = strtotime($comment['date']['created']); 4373011fb8bSMichael Klier } else { 4383011fb8bSMichael Klier $date = time(); 4393011fb8bSMichael Klier } 440f0fda08aSwikidesign 4413011fb8bSMichael Klier if ($date == -1) { 4423011fb8bSMichael Klier $date = time(); 4433011fb8bSMichael Klier } 4443011fb8bSMichael Klier 4456046f25cSwikidesign $cid = md5($comment['user']['id'].$date); // create a unique id 446f0fda08aSwikidesign 4473011fb8bSMichael Klier if (!is_array($data['comments'][$parent])) { 4483011fb8bSMichael Klier $parent = NULL; // invalid parent comment 4493011fb8bSMichael Klier } 450f0fda08aSwikidesign 451f0fda08aSwikidesign // render the comment 452f0fda08aSwikidesign $xhtml = $this->_render($comment['raw']); 453f0fda08aSwikidesign 454f0fda08aSwikidesign // fill in the new comment 455f0fda08aSwikidesign $data['comments'][$cid] = array( 4566046f25cSwikidesign 'user' => $comment['user'], 4576046f25cSwikidesign 'date' => array('created' => $date), 458f0fda08aSwikidesign 'show' => true, 4596046f25cSwikidesign 'raw' => $comment['raw'], 460f0fda08aSwikidesign 'xhtml' => $xhtml, 461f0fda08aSwikidesign 'parent' => $parent, 462a44bc9f7SMichael Klier 'replies' => array(), 463a44bc9f7SMichael Klier 'show' => $comment['show'] 464f0fda08aSwikidesign ); 465f0fda08aSwikidesign 4663011fb8bSMichael Klier if($comment['subscribe']) { 4673011fb8bSMichael Klier $mail = $comment['user']['mail']; 4683011fb8bSMichael Klier if($data['subscribers']) { 4693011fb8bSMichael Klier if(!$data['subscribers'][$mail]) { 4709881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4719881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4729881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4739881d835SMichael Klier } else { 4749881d835SMichael Klier // convert old style subscribers and set them active 4759881d835SMichael Klier if(!is_array($data['subscribers'][$mail])) { 4769881d835SMichael Klier $hash = $data['subscribers'][$mail]; 4779881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 4789881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 4799881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 4809881d835SMichael Klier } 4813011fb8bSMichael Klier } 4823011fb8bSMichael Klier } else { 4839881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4849881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4859881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4863011fb8bSMichael Klier } 4873011fb8bSMichael Klier } 4883011fb8bSMichael Klier 489f0fda08aSwikidesign // update parent comment 490f0fda08aSwikidesign if ($parent) $data['comments'][$parent]['replies'][] = $cid; 491f0fda08aSwikidesign 492f0fda08aSwikidesign // update the number of comments 493f0fda08aSwikidesign $data['number']++; 494f0fda08aSwikidesign 495f0fda08aSwikidesign // notify subscribers of the page 4968b42cefbSMichael Klier $data['comments'][$cid]['cid'] = $cid; 4973011fb8bSMichael Klier $this->_notify($data['comments'][$cid], $data['subscribers']); 498f0fda08aSwikidesign 4999881d835SMichael Klier // save the comment metadata file 5009881d835SMichael Klier io_saveFile($file, serialize($data)); 5019881d835SMichael Klier $this->_addLogEntry($date, $ID, 'cc', '', $cid); 5029881d835SMichael Klier 503a1d93126SGina Haeussge $this->_redirect($cid); 504f0fda08aSwikidesign return true; 505f0fda08aSwikidesign } 506f0fda08aSwikidesign 507f0fda08aSwikidesign /** 508f0fda08aSwikidesign * Saves the comment with the given ID and then displays all comments 509f0fda08aSwikidesign */ 510264b7327Swikidesign function _save($cids, $raw, $act = NULL) { 511a1ace3c9Swikidesign global $ID; 512f0fda08aSwikidesign 513757550e8SMichael Klier if(!$cids) return; // do nothing if we get no comment id 514757550e8SMichael Klier 5152ee3dca3Swikidesign if ($raw) { 5162ee3dca3Swikidesign global $TEXT; 5172ee3dca3Swikidesign 518f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 519f0fda08aSwikidesign $TEXT = $raw; 520f0fda08aSwikidesign 521f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 522f0fda08aSwikidesign if (checkwordblock()) { 523f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 524f0fda08aSwikidesign return false; 525f0fda08aSwikidesign } 526f0fda08aSwikidesign 527f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 5282ee3dca3Swikidesign } 529f0fda08aSwikidesign 530f0fda08aSwikidesign // get discussion meta file name 531f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 532f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 533f0fda08aSwikidesign 534573e23a1Swikidesign if (!is_array($cids)) $cids = array($cids); 535264b7327Swikidesign foreach ($cids as $cid) { 536264b7327Swikidesign 5376046f25cSwikidesign if (is_array($data['comments'][$cid]['user'])) { 5386046f25cSwikidesign $user = $data['comments'][$cid]['user']['id']; 5396046f25cSwikidesign $convert = false; 5406046f25cSwikidesign } else { 5416046f25cSwikidesign $user = $data['comments'][$cid]['user']; 5426046f25cSwikidesign $convert = true; 5436046f25cSwikidesign } 5446046f25cSwikidesign 545f0fda08aSwikidesign // someone else was trying to edit our comment -> abort 5463e02b3ffSwikidesign if (($user != $_SERVER['REMOTE_USER']) && (!auth_ismanager())) return false; 547f0fda08aSwikidesign 548f0fda08aSwikidesign $date = time(); 549f0fda08aSwikidesign 5506046f25cSwikidesign // need to convert to new format? 5516046f25cSwikidesign if ($convert) { 5526046f25cSwikidesign $data['comments'][$cid]['user'] = array( 5536046f25cSwikidesign 'id' => $user, 5546046f25cSwikidesign 'name' => $data['comments'][$cid]['name'], 5556046f25cSwikidesign 'mail' => $data['comments'][$cid]['mail'], 5566046f25cSwikidesign 'url' => $data['comments'][$cid]['url'], 5576046f25cSwikidesign 'address' => $data['comments'][$cid]['address'], 5586046f25cSwikidesign ); 5596046f25cSwikidesign $data['comments'][$cid]['date'] = array( 5606046f25cSwikidesign 'created' => $data['comments'][$cid]['date'] 5616046f25cSwikidesign ); 5626046f25cSwikidesign } 5636046f25cSwikidesign 564264b7327Swikidesign if ($act == 'toogle') { // toogle visibility 565f0fda08aSwikidesign $now = $data['comments'][$cid]['show']; 566f0fda08aSwikidesign $data['comments'][$cid]['show'] = !$now; 567f0fda08aSwikidesign $data['number'] = $this->_count($data); 568f0fda08aSwikidesign 569f0fda08aSwikidesign $type = ($data['comments'][$cid]['show'] ? 'sc' : 'hc'); 570f0fda08aSwikidesign 571264b7327Swikidesign } elseif ($act == 'show') { // show comment 572264b7327Swikidesign $data['comments'][$cid]['show'] = true; 573264b7327Swikidesign $data['number'] = $this->_count($data); 574264b7327Swikidesign 575573e23a1Swikidesign $type = 'sc'; // show comment 576264b7327Swikidesign 577264b7327Swikidesign } elseif ($act == 'hide') { // hide comment 578264b7327Swikidesign $data['comments'][$cid]['show'] = false; 579264b7327Swikidesign $data['number'] = $this->_count($data); 580264b7327Swikidesign 581573e23a1Swikidesign $type = 'hc'; // hide comment 582264b7327Swikidesign 583f0fda08aSwikidesign } elseif (!$raw) { // remove the comment 584efbe59d0Swikidesign $data['comments'] = $this->_removeComment($cid, $data['comments']); 585f0fda08aSwikidesign $data['number'] = $this->_count($data); 586f0fda08aSwikidesign 587573e23a1Swikidesign $type = 'dc'; // delete comment 588f0fda08aSwikidesign 589f0fda08aSwikidesign } else { // save changed comment 590f0fda08aSwikidesign $xhtml = $this->_render($raw); 591f0fda08aSwikidesign 592f0fda08aSwikidesign // now change the comment's content 5936046f25cSwikidesign $data['comments'][$cid]['date']['modified'] = $date; 5946046f25cSwikidesign $data['comments'][$cid]['raw'] = $raw; 595f0fda08aSwikidesign $data['comments'][$cid]['xhtml'] = $xhtml; 596f0fda08aSwikidesign 597573e23a1Swikidesign $type = 'ec'; // edit comment 598f0fda08aSwikidesign } 599264b7327Swikidesign } 600264b7327Swikidesign 601f0fda08aSwikidesign // save the comment metadata file 602f0fda08aSwikidesign io_saveFile($file, serialize($data)); 603f0fda08aSwikidesign $this->_addLogEntry($date, $ID, $type, '', $cid); 604f0fda08aSwikidesign 605a1d93126SGina Haeussge $this->_redirect($cid); 606f0fda08aSwikidesign return true; 607f0fda08aSwikidesign } 608f0fda08aSwikidesign 609f0fda08aSwikidesign /** 610efbe59d0Swikidesign * Recursive function to remove a comment 611efbe59d0Swikidesign */ 612efbe59d0Swikidesign function _removeComment($cid, $comments) { 613efbe59d0Swikidesign if (is_array($comments[$cid]['replies'])) { 614efbe59d0Swikidesign foreach ($comments[$cid]['replies'] as $rid) { 615efbe59d0Swikidesign $comments = $this->_removeComment($rid, $comments); 616efbe59d0Swikidesign } 617efbe59d0Swikidesign } 618efbe59d0Swikidesign unset($comments[$cid]); 619efbe59d0Swikidesign return $comments; 620efbe59d0Swikidesign } 621efbe59d0Swikidesign 622efbe59d0Swikidesign /** 623f0fda08aSwikidesign * Prints an individual comment 624f0fda08aSwikidesign */ 625f0fda08aSwikidesign function _print($cid, &$data, $parent = '', $reply = '', $visible = true) { 626f0fda08aSwikidesign 6272ee3dca3Swikidesign if (!isset($data['comments'][$cid])) return false; // comment was removed 628f0fda08aSwikidesign $comment = $data['comments'][$cid]; 629f0fda08aSwikidesign 630f0fda08aSwikidesign if (!is_array($comment)) return false; // corrupt datatype 631f0fda08aSwikidesign 632f0fda08aSwikidesign if ($comment['parent'] != $parent) return true; // reply to an other comment 633f0fda08aSwikidesign 634f0fda08aSwikidesign if (!$comment['show']) { // comment hidden 635a1ace3c9Swikidesign if (auth_ismanager()) $hidden = ' comment_hidden'; 636f0fda08aSwikidesign else return true; 6374a0a1bd2Swikidesign } else { 6384a0a1bd2Swikidesign $hidden = ''; 639f0fda08aSwikidesign } 640f0fda08aSwikidesign 641f1c6610eSpierre.spring // print the actual comment 6426b1c1263SMichael Klier $this->_print_comment($cid, $data, $parent, $reply, $visible, $hidden); 643f1c6610eSpierre.spring // replies to this comment entry? 644f1c6610eSpierre.spring $this->_print_replies($cid, $data, $reply, $visible); 645f1c6610eSpierre.spring // reply form 646f1c6610eSpierre.spring $this->_print_form($cid, $reply); 647f1c6610eSpierre.spring } 648f1c6610eSpierre.spring 649f1c6610eSpierre.spring function _print_comment($cid, &$data, $parent, $reply, $visible, $hidden) 650f1c6610eSpierre.spring { 651f1c6610eSpierre.spring global $conf, $lang, $ID, $HIGH; 652f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 653f1c6610eSpierre.spring 654f0fda08aSwikidesign // comment head with date and user data 6554a0a1bd2Swikidesign ptln('<div class="hentry'.$hidden.'">', 4); 6564a0a1bd2Swikidesign ptln('<div class="comment_head">', 6); 6571810ba9cSMichael Klier ptln('<a name="comment_'.$cid.'" id="comment_'.$cid.'"></a>', 8); 6584a0a1bd2Swikidesign $head = '<span class="vcard author">'; 659f0fda08aSwikidesign 6606046f25cSwikidesign // prepare variables 6616046f25cSwikidesign if (is_array($comment['user'])) { // new format 6626046f25cSwikidesign $user = $comment['user']['id']; 6636046f25cSwikidesign $name = $comment['user']['name']; 6646046f25cSwikidesign $mail = $comment['user']['mail']; 6656046f25cSwikidesign $url = $comment['user']['url']; 6666046f25cSwikidesign $address = $comment['user']['address']; 6676046f25cSwikidesign } else { // old format 6686046f25cSwikidesign $user = $comment['user']; 6696046f25cSwikidesign $name = $comment['name']; 6706046f25cSwikidesign $mail = $comment['mail']; 6716046f25cSwikidesign $url = $comment['url']; 6726046f25cSwikidesign $address = $comment['address']; 6736046f25cSwikidesign } 6746046f25cSwikidesign if (is_array($comment['date'])) { // new format 6756046f25cSwikidesign $created = $comment['date']['created']; 6766046f25cSwikidesign $modified = $comment['date']['modified']; 6776046f25cSwikidesign } else { // old format 6786046f25cSwikidesign $created = $comment['date']; 6796046f25cSwikidesign $modified = $comment['edited']; 6806046f25cSwikidesign } 6816046f25cSwikidesign 6820ff5ab97SMichael Klier // show username or real name? 6830ff5ab97SMichael Klier if ((!$this->getConf('userealname')) && ($user)) { 6840ff5ab97SMichael Klier $showname = $user; 6850ff5ab97SMichael Klier } else { 6860ff5ab97SMichael Klier $showname = $name; 6870ff5ab97SMichael Klier } 6880ff5ab97SMichael Klier 689ce7b17cfSwikidesign // show avatar image? 690f1c6610eSpierre.spring if ($this->_use_avatar()) { 69112ca6034SMichael Klier $user_data['name'] = $name; 69212ca6034SMichael Klier $user_data['user'] = $user; 69312ca6034SMichael Klier $user_data['mail'] = $mail; 69412ca6034SMichael Klier $avatar = $this->avatar->getXHTML($user_data, $name, 'left'); 695db963b87SMichael Klier if($avatar) $head .= $avatar; 696f0fda08aSwikidesign } 697f0fda08aSwikidesign 6986046f25cSwikidesign if ($this->getConf('linkemail') && $mail) { 6990ff5ab97SMichael Klier $head .= $this->email($mail, $showname, 'email fn'); 7006046f25cSwikidesign } elseif ($url) { 7010ff5ab97SMichael Klier $head .= $this->external_link($this->_checkURL($url), $showname, 'urlextern url fn'); 702f0fda08aSwikidesign } else { 7030ff5ab97SMichael Klier $head .= '<span class="fn">'.$showname.'</span>'; 704f0fda08aSwikidesign } 7054a0a1bd2Swikidesign if ($address) $head .= ', <span class="adr">'.$address.'</span>'; 7064a0a1bd2Swikidesign $head .= '</span>, '. 707f014bc86SMichael Klier '<abbr class="published" title="'.strftime('%Y-%m-%dT%H:%M:%SZ', $created).'">'. 708f014bc86SMichael Klier strftime($conf['dformat'], $created).'</abbr>'; 7094a0a1bd2Swikidesign if ($comment['edited']) $head .= ' (<abbr class="updated" title="'. 710f014bc86SMichael Klier strftime('%Y-%m-%dT%H:%M:%SZ', $modified).'">'.strftime($conf['dformat'], $modified). 7116046f25cSwikidesign '</abbr>)'; 712f014bc86SMichael Klier ptln($head, 8); 7134a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_head" 714f0fda08aSwikidesign 715f0fda08aSwikidesign // main comment content 7164a0a1bd2Swikidesign ptln('<div class="comment_body entry-content"'. 717f1c6610eSpierre.spring ($this->getConf('useavatar') ? $this->_get_style() : '').'>', 6); 71815cdad37Spierre.spring echo ($HIGH?html_hilight($comment['xhtml'],$HIGH):$comment['xhtml']).DOKU_LF; 7194a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_body" 720f0fda08aSwikidesign 721f0fda08aSwikidesign if ($visible) { 7221184c36aSwikidesign ptln('<div class="comment_buttons">', 6); 723f0fda08aSwikidesign 724f0fda08aSwikidesign // show reply button? 725f1c4aa1aSwikidesign if (($data['status'] == 1) && !$reply && $comment['show'] 72631aab30eSGina Haeussge && ($this->getConf('allowguests') || $_SERVER['REMOTE_USER']) && $this->getConf('usethreading')) 7271e46d176Swikidesign $this->_button($cid, $this->getLang('btn_reply'), 'reply', true); 728f0fda08aSwikidesign 7291184c36aSwikidesign // show edit, show/hide and delete button? 7301184c36aSwikidesign if ((($user == $_SERVER['REMOTE_USER']) && ($user != '')) || (auth_ismanager())) { 7311e46d176Swikidesign $this->_button($cid, $lang['btn_secedit'], 'edit', true); 7321184c36aSwikidesign $label = ($comment['show'] ? $this->getLang('btn_hide') : $this->getLang('btn_show')); 7331184c36aSwikidesign $this->_button($cid, $label, 'toogle'); 7341184c36aSwikidesign $this->_button($cid, $lang['btn_delete'], 'delete'); 735f0fda08aSwikidesign } 7361184c36aSwikidesign ptln('</div>', 6); // class="comment_buttons" 7371184c36aSwikidesign } 7381184c36aSwikidesign ptln('</div>', 4); // class="hentry" 739f0fda08aSwikidesign } 740f0fda08aSwikidesign 741f1c6610eSpierre.spring function _print_form($cid, $reply) 742f1c6610eSpierre.spring { 74331aab30eSGina Haeussge if ($this->getConf('usethreading') && $reply == $cid) { 7444a0a1bd2Swikidesign ptln('<div class="comment_replies">', 4); 745f0fda08aSwikidesign $this->_form('', 'add', $cid); 7464a0a1bd2Swikidesign ptln('</div>', 4); // class="comment_replies" 747f0fda08aSwikidesign } 748f0fda08aSwikidesign } 749f0fda08aSwikidesign 750f1c6610eSpierre.spring function _print_replies($cid, &$data, $reply, &$visible) 751f1c6610eSpierre.spring { 752f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 753f1c6610eSpierre.spring if (!count($comment['replies'])) { 754f1c6610eSpierre.spring return; 755f1c6610eSpierre.spring } 756f1c6610eSpierre.spring ptln('<div class="comment_replies"'.$this->_get_style().'>', 4); 757f1c6610eSpierre.spring $visible = ($comment['show'] && $visible); 758f1c6610eSpierre.spring foreach ($comment['replies'] as $rid) { 759f1c6610eSpierre.spring $this->_print($rid, $data, $cid, $reply, $visible); 760f1c6610eSpierre.spring } 761f1c6610eSpierre.spring ptln('</div>', 4); 762f1c6610eSpierre.spring } 763f1c6610eSpierre.spring 764f1c6610eSpierre.spring function _use_avatar() 765f1c6610eSpierre.spring { 766f1c6610eSpierre.spring if (is_null($this->use_avatar)) { 767f1c6610eSpierre.spring $this->use_avatar = $this->getConf('useavatar') 768f1c6610eSpierre.spring && (!plugin_isdisabled('avatar')) 769f1c6610eSpierre.spring && ($this->avatar =& plugin_load('helper', 'avatar')); 770f1c6610eSpierre.spring } 771f1c6610eSpierre.spring return $this->use_avatar; 772f1c6610eSpierre.spring } 773f1c6610eSpierre.spring 774f1c6610eSpierre.spring function _get_style() 775f1c6610eSpierre.spring { 776f1c6610eSpierre.spring if (is_null($this->style)){ 777f1c6610eSpierre.spring if ($this->_use_avatar()) { 778f1c6610eSpierre.spring $this->style = ' style="margin-left: '.($this->avatar->getConf('size') + 14).'px;"'; 779f1c6610eSpierre.spring } else { 780f1c6610eSpierre.spring $this->style = ' style="margin-left: 20px;"'; 781f1c6610eSpierre.spring } 782f1c6610eSpierre.spring } 783f1c6610eSpierre.spring return $this->style; 784f1c6610eSpierre.spring } 785f1c6610eSpierre.spring 786f0fda08aSwikidesign /** 787f0fda08aSwikidesign * Outputs the comment form 788f0fda08aSwikidesign */ 789f0fda08aSwikidesign function _form($raw = '', $act = 'add', $cid = NULL) { 7903011fb8bSMichael Klier global $lang; 7913011fb8bSMichael Klier global $conf; 7923011fb8bSMichael Klier global $ID; 7933011fb8bSMichael Klier global $INFO; 794f0fda08aSwikidesign 795f0fda08aSwikidesign // not for unregistered users when guest comments aren't allowed 796f0fda08aSwikidesign if (!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) return false; 797f0fda08aSwikidesign 7981ba72c23Swikidesign // fill $raw with $_REQUEST['text'] if it's empty (for failed CAPTCHA check) 7991ba72c23Swikidesign if (!$raw && ($_REQUEST['comment'] == 'show')) $raw = $_REQUEST['text']; 800f0fda08aSwikidesign ?> 8015ef1705fSiLoveiDo 802f0fda08aSwikidesign <div class="comment_form"> 803bea69264SMichael Klier <form id="discussion__comment_form" method="post" action="<?php echo script() ?>" accept-charset="<?php echo $lang['encoding'] ?>"> 804f0fda08aSwikidesign <div class="no"> 805f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 80661437513Swikidesign <input type="hidden" name="do" value="show" /> 807f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 808530693fbSMichael Klier <?php 809f0fda08aSwikidesign // for adding a comment 810f0fda08aSwikidesign if ($act == 'add') { 811f0fda08aSwikidesign ?> 812f0fda08aSwikidesign <input type="hidden" name="reply" value="<?php echo $cid ?>" /> 813f0fda08aSwikidesign <?php 81420c152acSMichael Klier // for guest/adminimport: show name, e-mail and subscribe to comments fields 81520c152acSMichael Klier if(!$_SERVER['REMOTE_USER'] or ($this->getConf('adminimport') && auth_ismanager())) { 816f0fda08aSwikidesign ?> 817f0fda08aSwikidesign <input type="hidden" name="user" value="<?php echo clientIP() ?>" /> 818f0fda08aSwikidesign <div class="comment_name"> 819f0fda08aSwikidesign <label class="block" for="discussion__comment_name"> 820f0fda08aSwikidesign <span><?php echo $lang['fullname'] ?>:</span> 82137e3c825SMichael Klier <input type="text" class="edit<?php if($_REQUEST['comment'] == 'add' && empty($_REQUEST['name'])) echo ' error'?>" name="name" id="discussion__comment_name" size="50" tabindex="1" value="<?php echo hsc($_REQUEST['name'])?>" /> 822f0fda08aSwikidesign </label> 823f0fda08aSwikidesign </div> 824f0fda08aSwikidesign <div class="comment_mail"> 825f0fda08aSwikidesign <label class="block" for="discussion__comment_mail"> 826f0fda08aSwikidesign <span><?php echo $lang['email'] ?>:</span> 82737e3c825SMichael Klier <input type="text" class="edit<?php if($_REQUEST['comment'] == 'add' && empty($_REQUEST['mail'])) echo ' error'?>" name="mail" id="discussion__comment_mail" size="50" tabindex="2" value="<?php echo hsc($_REQUEST['mail'])?>" /> 828f0fda08aSwikidesign </label> 829f0fda08aSwikidesign </div> 830f0fda08aSwikidesign <?php 831f0fda08aSwikidesign } 832f0fda08aSwikidesign 833f0fda08aSwikidesign // allow entering an URL 834f0fda08aSwikidesign if ($this->getConf('urlfield')) { 835f0fda08aSwikidesign ?> 836f0fda08aSwikidesign <div class="comment_url"> 837f0fda08aSwikidesign <label class="block" for="discussion__comment_url"> 838f0fda08aSwikidesign <span><?php echo $this->getLang('url') ?>:</span> 839e7c760b3Swikidesign <input type="text" class="edit" name="url" id="discussion__comment_url" size="50" tabindex="3" value="<?php echo hsc($_REQUEST['url'])?>" /> 840f0fda08aSwikidesign </label> 841f0fda08aSwikidesign </div> 842f0fda08aSwikidesign <?php 843f0fda08aSwikidesign } 844f0fda08aSwikidesign 845f0fda08aSwikidesign // allow entering an address 846f0fda08aSwikidesign if ($this->getConf('addressfield')) { 847f0fda08aSwikidesign ?> 848f0fda08aSwikidesign <div class="comment_address"> 849f0fda08aSwikidesign <label class="block" for="discussion__comment_address"> 850f0fda08aSwikidesign <span><?php echo $this->getLang('address') ?>:</span> 851e7c760b3Swikidesign <input type="text" class="edit" name="address" id="discussion__comment_address" size="50" tabindex="4" value="<?php echo hsc($_REQUEST['address'])?>" /> 852f0fda08aSwikidesign </label> 853f0fda08aSwikidesign </div> 854f0fda08aSwikidesign <?php 855f0fda08aSwikidesign } 856f0fda08aSwikidesign 857f0fda08aSwikidesign // allow setting the comment date 858a1ace3c9Swikidesign if ($this->getConf('adminimport') && (auth_ismanager())) { 859f0fda08aSwikidesign ?> 860f0fda08aSwikidesign <div class="comment_date"> 861f0fda08aSwikidesign <label class="block" for="discussion__comment_date"> 862f0fda08aSwikidesign <span><?php echo $this->getLang('date') ?>:</span> 863f0fda08aSwikidesign <input type="text" class="edit" name="date" id="discussion__comment_date" size="50" /> 864f0fda08aSwikidesign </label> 865f0fda08aSwikidesign </div> 866f0fda08aSwikidesign <?php 867f0fda08aSwikidesign } 868f0fda08aSwikidesign 869f0fda08aSwikidesign // for saving a comment 870f0fda08aSwikidesign } else { 871f0fda08aSwikidesign ?> 872f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 873f0fda08aSwikidesign <?php 874f0fda08aSwikidesign } 875f0fda08aSwikidesign ?> 876f0fda08aSwikidesign <div class="comment_text"> 8771de52da1SMichael Klier <div id="discussion__comment_toolbar"> 8781de52da1SMichael Klier <?php echo $this->getLang('entercomment')?> 8791de52da1SMichael Klier <?php if($this->getLang('wikisyntaxok')) echo ', ' . $this->getLang('wikisyntax') . ':';?> 8801de52da1SMichael Klier </div> 88137e3c825SMichael Klier <textarea class="edit<?php if($_REQUEST['comment'] == 'add' && empty($_REQUEST['text'])) echo ' error'?>" name="text" cols="80" rows="10" id="discussion__comment_text" tabindex="5"><?php 88237e3c825SMichael Klier if($raw) { 88337e3c825SMichael Klier echo formText($raw); 88437e3c825SMichael Klier } else { 88537e3c825SMichael Klier echo $_REQUEST['text']; 88637e3c825SMichael Klier } 88737e3c825SMichael Klier ?></textarea> 888f0fda08aSwikidesign </div> 889e7c760b3Swikidesign <?php //bad and dirty event insert hook 890e7c760b3Swikidesign $evdata = array('writable' => true); 891e7c760b3Swikidesign trigger_event('HTML_EDITFORM_INJECTION', $evdata); 892e7c760b3Swikidesign ?> 893bea69264SMichael 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" /> 8940c3df1b0SMichael Klier <input class="button comment_preview_button" id="discussion__btn_preview" type="button" name="preview" accesskey="p" value="<?php echo $lang['btn_preview'] ?>" title="<?php echo $lang['btn_preview']?> [P]" /> 8953011fb8bSMichael Klier 896189071d2SMichael Klier <?php if((!$_SERVER['REMOTE_USER'] || $_SERVER['REMOTE_USER'] && !$conf['subscribers']) && $this->getConf('subscribe')) { ?> 8973011fb8bSMichael Klier <div class="comment_subscribe"> 89811b1339eSGina Haeussge <input type="checkbox" id="discussion__comment_subscribe" name="subscribe" tabindex="6" /> 8993011fb8bSMichael Klier <label class="block" for="discussion__comment_subscribe"> 9003011fb8bSMichael Klier <span><?php echo $this->getLang('subscribe') ?></span> 9013011fb8bSMichael Klier </label> 9023011fb8bSMichael Klier </div> 9033011fb8bSMichael Klier <?php } ?> 9043011fb8bSMichael Klier 9053011fb8bSMichael Klier <div class="clearer"></div> 906ed4846efSMichael Klier <div id="discussion__comment_preview"> </div> 907f0fda08aSwikidesign </div> 908f0fda08aSwikidesign </form> 909f0fda08aSwikidesign </div> 910f0fda08aSwikidesign <?php 911f0fda08aSwikidesign if ($this->getConf('usecocomment')) echo $this->_coComment(); 912f0fda08aSwikidesign } 913f0fda08aSwikidesign 914f0fda08aSwikidesign /** 915f0fda08aSwikidesign * Adds a javascript to interact with coComments 916f0fda08aSwikidesign */ 917f0fda08aSwikidesign function _coComment() { 9182b18adb9SMichael Klier global $ID; 9192b18adb9SMichael Klier global $conf; 9202b18adb9SMichael Klier global $INFO; 921f0fda08aSwikidesign 922f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 923f0fda08aSwikidesign 924f0fda08aSwikidesign ?> 925f0fda08aSwikidesign <script type="text/javascript"><!--//--><![CDATA[//><!-- 926f0fda08aSwikidesign var blogTool = "DokuWiki"; 927f0fda08aSwikidesign var blogURL = "<?php echo DOKU_URL ?>"; 928f0fda08aSwikidesign var blogTitle = "<?php echo $conf['title'] ?>"; 929f0fda08aSwikidesign var postURL = "<?php echo wl($ID, '', true) ?>"; 930f0fda08aSwikidesign var postTitle = "<?php echo tpl_pagetitle($ID, true) ?>"; 931f0fda08aSwikidesign <?php 932f0fda08aSwikidesign if ($user) { 933f0fda08aSwikidesign ?> 934f0fda08aSwikidesign var commentAuthor = "<?php echo $INFO['userinfo']['name'] ?>"; 935f0fda08aSwikidesign <?php 936f0fda08aSwikidesign } else { 937f0fda08aSwikidesign ?> 938f0fda08aSwikidesign var commentAuthorFieldName = "name"; 939f0fda08aSwikidesign <?php 940f0fda08aSwikidesign } 941f0fda08aSwikidesign ?> 942f0fda08aSwikidesign var commentAuthorLoggedIn = <?php echo ($user ? 'true' : 'false') ?>; 943f0fda08aSwikidesign var commentFormID = "discussion__comment_form"; 944f0fda08aSwikidesign var commentTextFieldName = "text"; 945f0fda08aSwikidesign var commentButtonName = "submit"; 946f0fda08aSwikidesign var cocomment_force = false; 947f0fda08aSwikidesign //--><!]]></script> 948f0fda08aSwikidesign <script type="text/javascript" src="http://www.cocomment.com/js/cocomment.js"> 949f0fda08aSwikidesign </script> 950f0fda08aSwikidesign <?php 951f0fda08aSwikidesign } 952f0fda08aSwikidesign 953f0fda08aSwikidesign /** 954f0fda08aSwikidesign * General button function 955f0fda08aSwikidesign */ 9561e46d176Swikidesign function _button($cid, $label, $act, $jump = false) { 957f0fda08aSwikidesign global $ID; 9585ef1705fSiLoveiDo 9591e46d176Swikidesign $anchor = ($jump ? '#discussion__comment_form' : '' ); 960f0fda08aSwikidesign 961f0fda08aSwikidesign ?> 9626d1f4f20SGina Haeussge <form class="button discussion__<?php echo $act?>" method="get" action="<?php echo script().$anchor ?>"> 963f0fda08aSwikidesign <div class="no"> 964f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 96561437513Swikidesign <input type="hidden" name="do" value="show" /> 966f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 967f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 968f0fda08aSwikidesign <input type="submit" value="<?php echo $label ?>" class="button" title="<?php echo $label ?>" /> 969f0fda08aSwikidesign </div> 970f0fda08aSwikidesign </form> 971f0fda08aSwikidesign <?php 972f0fda08aSwikidesign return true; 973f0fda08aSwikidesign } 974f0fda08aSwikidesign 975f0fda08aSwikidesign /** 976f0fda08aSwikidesign * Adds an entry to the comments changelog 977f0fda08aSwikidesign * 978f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 979f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 980f0fda08aSwikidesign */ 981f0fda08aSwikidesign function _addLogEntry($date, $id, $type = 'cc', $summary = '', $extra = '') { 982f0fda08aSwikidesign global $conf; 983f0fda08aSwikidesign 984f0fda08aSwikidesign $changelog = $conf['metadir'].'/_comments.changes'; 985f0fda08aSwikidesign 986f0fda08aSwikidesign if(!$date) $date = time(); //use current time if none supplied 987f0fda08aSwikidesign $remote = $_SERVER['REMOTE_ADDR']; 988f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 989f0fda08aSwikidesign 990f0fda08aSwikidesign $strip = array("\t", "\n"); 991f0fda08aSwikidesign $logline = array( 992f0fda08aSwikidesign 'date' => $date, 993f0fda08aSwikidesign 'ip' => $remote, 994f0fda08aSwikidesign 'type' => str_replace($strip, '', $type), 995f0fda08aSwikidesign 'id' => $id, 996f0fda08aSwikidesign 'user' => $user, 997f0fda08aSwikidesign 'sum' => str_replace($strip, '', $summary), 998f0fda08aSwikidesign 'extra' => str_replace($strip, '', $extra) 999f0fda08aSwikidesign ); 1000f0fda08aSwikidesign 1001f0fda08aSwikidesign // add changelog line 1002f0fda08aSwikidesign $logline = implode("\t", $logline)."\n"; 1003f0fda08aSwikidesign io_saveFile($changelog, $logline, true); //global changelog cache 1004f0fda08aSwikidesign $this->_trimRecentCommentsLog($changelog); 100577a22ba2Swikidesign 100677a22ba2Swikidesign // tell the indexer to re-index the page 100777a22ba2Swikidesign @unlink(metaFN($id, '.indexed')); 1008f0fda08aSwikidesign } 1009f0fda08aSwikidesign 1010f0fda08aSwikidesign /** 1011f0fda08aSwikidesign * Trims the recent comments cache to the last $conf['changes_days'] recent 1012f0fda08aSwikidesign * changes or $conf['recent'] items, which ever is larger. 1013f0fda08aSwikidesign * The trimming is only done once a day. 1014f0fda08aSwikidesign * 1015f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 1016f0fda08aSwikidesign */ 1017f0fda08aSwikidesign function _trimRecentCommentsLog($changelog) { 1018f0fda08aSwikidesign global $conf; 1019f0fda08aSwikidesign 1020f0fda08aSwikidesign if (@file_exists($changelog) && 1021f0fda08aSwikidesign (filectime($changelog) + 86400) < time() && 1022f0fda08aSwikidesign !@file_exists($changelog.'_tmp')) { 1023f0fda08aSwikidesign 1024f0fda08aSwikidesign io_lock($changelog); 1025f0fda08aSwikidesign $lines = file($changelog); 1026f0fda08aSwikidesign if (count($lines)<$conf['recent']) { 1027f0fda08aSwikidesign // nothing to trim 1028f0fda08aSwikidesign io_unlock($changelog); 1029f0fda08aSwikidesign return true; 1030f0fda08aSwikidesign } 1031f0fda08aSwikidesign 1032f0fda08aSwikidesign io_saveFile($changelog.'_tmp', ''); // presave tmp as 2nd lock 1033f0fda08aSwikidesign $trim_time = time() - $conf['recent_days']*86400; 1034f0fda08aSwikidesign $out_lines = array(); 1035f0fda08aSwikidesign 1036e49085a2SMichael Klier $num = count($lines); 1037e49085a2SMichael Klier for ($i=0; $i<$num; $i++) { 1038f0fda08aSwikidesign $log = parseChangelogLine($lines[$i]); 1039f0fda08aSwikidesign if ($log === false) continue; // discard junk 1040f0fda08aSwikidesign if ($log['date'] < $trim_time) { 1041f0fda08aSwikidesign $old_lines[$log['date'].".$i"] = $lines[$i]; // keep old lines for now (append .$i to prevent key collisions) 1042f0fda08aSwikidesign } else { 1043f0fda08aSwikidesign $out_lines[$log['date'].".$i"] = $lines[$i]; // definitely keep these lines 1044f0fda08aSwikidesign } 1045f0fda08aSwikidesign } 1046f0fda08aSwikidesign 1047f0fda08aSwikidesign // sort the final result, it shouldn't be necessary, 1048f0fda08aSwikidesign // however the extra robustness in making the changelog cache self-correcting is worth it 1049f0fda08aSwikidesign ksort($out_lines); 1050f0fda08aSwikidesign $extra = $conf['recent'] - count($out_lines); // do we need extra lines do bring us up to minimum 1051f0fda08aSwikidesign if ($extra > 0) { 1052f0fda08aSwikidesign ksort($old_lines); 1053f0fda08aSwikidesign $out_lines = array_merge(array_slice($old_lines,-$extra),$out_lines); 1054f0fda08aSwikidesign } 1055f0fda08aSwikidesign 1056f0fda08aSwikidesign // save trimmed changelog 1057f0fda08aSwikidesign io_saveFile($changelog.'_tmp', implode('', $out_lines)); 1058f0fda08aSwikidesign @unlink($changelog); 1059f0fda08aSwikidesign if (!rename($changelog.'_tmp', $changelog)) { 1060f0fda08aSwikidesign // rename failed so try another way... 1061f0fda08aSwikidesign io_unlock($changelog); 1062f0fda08aSwikidesign io_saveFile($changelog, implode('', $out_lines)); 1063f0fda08aSwikidesign @unlink($changelog.'_tmp'); 1064f0fda08aSwikidesign } else { 1065f0fda08aSwikidesign io_unlock($changelog); 1066f0fda08aSwikidesign } 1067f0fda08aSwikidesign return true; 1068f0fda08aSwikidesign } 1069f0fda08aSwikidesign } 1070f0fda08aSwikidesign 1071f0fda08aSwikidesign /** 1072f0fda08aSwikidesign * Sends a notify mail on new comment 1073f0fda08aSwikidesign * 1074f0fda08aSwikidesign * @param array $comment data array of the new comment 1075f0fda08aSwikidesign * 1076f0fda08aSwikidesign * @author Andreas Gohr <andi@splitbrain.org> 1077f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 1078f0fda08aSwikidesign */ 10799881d835SMichael Klier function _notify($comment, &$subscribers) { 1080f0fda08aSwikidesign global $conf; 1081f0fda08aSwikidesign global $ID; 1082bd15cd45SAndy Webber global $INFO; 1083f0fda08aSwikidesign 10849881d835SMichael Klier $notify_text = io_readfile($this->localfn('subscribermail')); 10859881d835SMichael Klier $confirm_text = io_readfile($this->localfn('confirmsubscribe')); 10869881d835SMichael Klier $subject_notify = '['.$conf['title'].'] '.$this->getLang('mail_newcomment'); 10879881d835SMichael Klier $subject_subscribe = '['.$conf['title'].'] '.$this->getLang('subscribe'); 1088bd15cd45SAndy Webber $from = $conf['mailfrom']; 1089bd15cd45SAndy Webber $from = str_replace('@USER@',$_SERVER['REMOTE_USER'],$from); 1090bd15cd45SAndy Webber $from = str_replace('@NAME@',$INFO['userinfo']['name'],$from); 1091bd15cd45SAndy Webber $from = str_replace('@MAIL@',$INFO['userinfo']['mail'],$from); 1092f0fda08aSwikidesign 10936046f25cSwikidesign $search = array( 10946046f25cSwikidesign '@PAGE@', 10956046f25cSwikidesign '@TITLE@', 10966046f25cSwikidesign '@DATE@', 10976046f25cSwikidesign '@NAME@', 10986046f25cSwikidesign '@TEXT@', 10998b42cefbSMichael Klier '@COMMENTURL@', 11006046f25cSwikidesign '@UNSUBSCRIBE@', 11016046f25cSwikidesign '@DOKUWIKIURL@', 11026046f25cSwikidesign ); 11033011fb8bSMichael Klier 11043011fb8bSMichael Klier // notify page subscribers 11058c6e87cfSMichael Klier if ($conf['subscribers'] || $conf['notify']) { 1106189071d2SMichael Klier $list = explode(',', subscriber_addresslist($ID)); 11078c6e87cfSMichael Klier $to = (!empty($conf['notify'])) ? $conf['notify'] : array_pop($list); 1108189071d2SMichael Klier $bcc = implode(',', $list); 11093011fb8bSMichael Klier 11106046f25cSwikidesign $replace = array( 11116046f25cSwikidesign $ID, 11126046f25cSwikidesign $conf['title'], 1113f014bc86SMichael Klier strftime($conf['dformat'], $comment['date']['created']), 11146046f25cSwikidesign $comment['user']['name'], 11156046f25cSwikidesign $comment['raw'], 11161810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 11176046f25cSwikidesign wl($ID, 'do=unsubscribe', true, '&'), 11186046f25cSwikidesign DOKU_URL, 11196046f25cSwikidesign ); 1120f0fda08aSwikidesign 11219881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 1122bd15cd45SAndy Webber mail_send($to, $subject_notify, $body, $from, '', $bcc); 11233011fb8bSMichael Klier } 1124f0fda08aSwikidesign 11253011fb8bSMichael Klier // notify comment subscribers 11263011fb8bSMichael Klier if (!empty($subscribers)) { 11273011fb8bSMichael Klier 11289881d835SMichael Klier foreach($subscribers as $mail => $data) { 11293011fb8bSMichael Klier $to = $mail; 11303011fb8bSMichael Klier 11319881d835SMichael Klier if($data['active']) { 11323011fb8bSMichael Klier $replace = array( 11333011fb8bSMichael Klier $ID, 11343011fb8bSMichael Klier $conf['title'], 11353011fb8bSMichael Klier strftime($conf['dformat'], $comment['date']['created']), 11363011fb8bSMichael Klier $comment['user']['name'], 11373011fb8bSMichael Klier $comment['raw'], 11381810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 1139b2f2e866SMichael Klier wl($ID, 'do=discussion_unsubscribe&hash=' . $data['hash'], true, '&'), 11403011fb8bSMichael Klier DOKU_URL, 11413011fb8bSMichael Klier ); 11423011fb8bSMichael Klier 11439881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 1144bd15cd45SAndy Webber mail_send($to, $subject_notify, $body, $from); 11459881d835SMichael Klier } elseif(!$data['active'] && !$data['confirmsent']) { 11469881d835SMichael Klier $search = array( 11479881d835SMichael Klier '@PAGE@', 11489881d835SMichael Klier '@TITLE@', 11499881d835SMichael Klier '@SUBSCRIBE@', 11509881d835SMichael Klier '@DOKUWIKIURL@', 11519881d835SMichael Klier ); 11529881d835SMichael Klier $replace = array( 11539881d835SMichael Klier $ID, 11549881d835SMichael Klier $conf['title'], 1155b2f2e866SMichael Klier wl($ID, 'do=discussion_confirmsubscribe&hash=' . $data['hash'], true, '&'), 11569881d835SMichael Klier DOKU_URL, 11579881d835SMichael Klier ); 11589881d835SMichael Klier 11599881d835SMichael Klier $body = str_replace($search, $replace, $confirm_text); 1160bd15cd45SAndy Webber mail_send($to, $subject_subscribe, $body, $from); 11619881d835SMichael Klier $subscribers[$mail]['confirmsent'] = true; 11629881d835SMichael Klier } 11633011fb8bSMichael Klier } 11643011fb8bSMichael Klier } 1165f0fda08aSwikidesign } 1166f0fda08aSwikidesign 1167f0fda08aSwikidesign /** 1168f0fda08aSwikidesign * Counts the number of visible comments 1169f0fda08aSwikidesign */ 1170f0fda08aSwikidesign function _count($data) { 1171f0fda08aSwikidesign $number = 0; 1172f0fda08aSwikidesign foreach ($data['comments'] as $cid => $comment) { 1173f0fda08aSwikidesign if ($comment['parent']) continue; 1174f0fda08aSwikidesign if (!$comment['show']) continue; 1175f0fda08aSwikidesign $number++; 1176f0fda08aSwikidesign $rids = $comment['replies']; 1177f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1178f0fda08aSwikidesign } 1179f0fda08aSwikidesign return $number; 1180f0fda08aSwikidesign } 1181f0fda08aSwikidesign 1182f0fda08aSwikidesign function _countReplies(&$data, $rids) { 1183f0fda08aSwikidesign $number = 0; 1184f0fda08aSwikidesign foreach ($rids as $rid) { 11852ee3dca3Swikidesign if (!isset($data['comments'][$rid])) continue; // reply was removed 1186f0fda08aSwikidesign if (!$data['comments'][$rid]['show']) continue; 1187f0fda08aSwikidesign $number++; 1188f0fda08aSwikidesign $rids = $data['comments'][$rid]['replies']; 1189f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1190f0fda08aSwikidesign } 1191f0fda08aSwikidesign return $number; 1192f0fda08aSwikidesign } 1193f0fda08aSwikidesign 1194f0fda08aSwikidesign /** 1195f0fda08aSwikidesign * Renders the comment text 1196f0fda08aSwikidesign */ 1197f0fda08aSwikidesign function _render($raw) { 1198f0fda08aSwikidesign if ($this->getConf('wikisyntaxok')) { 1199f0fda08aSwikidesign $xhtml = $this->render($raw); 1200f0fda08aSwikidesign } else { // wiki syntax not allowed -> just encode special chars 120187bb4e97SMichael Klier $xhtml = hsc(trim($raw)); 120287bb4e97SMichael Klier $xhtml = str_replace("\n", '<br />', $xhtml); 1203f0fda08aSwikidesign } 1204f0fda08aSwikidesign return $xhtml; 1205f0fda08aSwikidesign } 1206f0fda08aSwikidesign 1207f0fda08aSwikidesign /** 1208479dd10fSwikidesign * Finds out whether there is a discussion section for the current page 1209479dd10fSwikidesign */ 1210e6a89be2Swikidesign function _hasDiscussion(&$title) { 1211b2ac3b3bSwikidesign global $ID; 12124a0a1bd2Swikidesign 1213479dd10fSwikidesign $cfile = metaFN($ID, '.comments'); 1214479dd10fSwikidesign 1215479dd10fSwikidesign if (!@file_exists($cfile)) { 12162b18adb9SMichael Klier if ($this->getConf('automatic')) { 12172b18adb9SMichael Klier return true; 12182b18adb9SMichael Klier } else { 12192b18adb9SMichael Klier return false; 12202b18adb9SMichael Klier } 1221479dd10fSwikidesign } 1222479dd10fSwikidesign 1223479dd10fSwikidesign $comments = unserialize(io_readFile($cfile, false)); 1224479dd10fSwikidesign 122507c376bbSwikidesign if ($comments['title']) $title = hsc($comments['title']); 1226479dd10fSwikidesign $num = $comments['number']; 1227479dd10fSwikidesign if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return false; 1228479dd10fSwikidesign else return true; 1229479dd10fSwikidesign } 1230479dd10fSwikidesign 1231479dd10fSwikidesign /** 1232e7c760b3Swikidesign * Creates a new thread page 1233e7c760b3Swikidesign */ 12346046f25cSwikidesign function _newThread() { 123561437513Swikidesign global $ID, $INFO; 1236f0fda08aSwikidesign 12371ea794e5Swikidesign $ns = cleanID($_REQUEST['ns']); 1238f0fda08aSwikidesign $title = str_replace(':', '', $_REQUEST['title']); 12392e80cd5fSwikidesign $back = $ID; 12402e80cd5fSwikidesign $ID = ($ns ? $ns.':' : '').cleanID($title); 12412e80cd5fSwikidesign $INFO = pageinfo(); 1242f0fda08aSwikidesign 1243f0fda08aSwikidesign // check if we are allowed to create this file 12442e80cd5fSwikidesign if ($INFO['perm'] >= AUTH_CREATE) { 1245f0fda08aSwikidesign 1246f0fda08aSwikidesign //check if locked by anyone - if not lock for my self 12472e80cd5fSwikidesign if ($INFO['locked']) return 'locked'; 12482e80cd5fSwikidesign else lock($ID); 1249f0fda08aSwikidesign 1250f0fda08aSwikidesign // prepare the new thread file with default stuff 12512e80cd5fSwikidesign if (!@file_exists($INFO['filepath'])) { 1252f0fda08aSwikidesign global $TEXT; 1253f0fda08aSwikidesign 12542e80cd5fSwikidesign $TEXT = pageTemplate(array(($ns ? $ns.':' : '').$title)); 12551433886fSwikidesign if (!$TEXT) { 125661437513Swikidesign $data = array('id' => $ID, 'ns' => $ns, 'title' => $title, 'back' => $back); 125761437513Swikidesign $TEXT = $this->_pageTemplate($data); 12582e80cd5fSwikidesign } 12592e80cd5fSwikidesign return 'preview'; 1260f0fda08aSwikidesign } else { 12612e80cd5fSwikidesign return 'edit'; 1262f0fda08aSwikidesign } 1263f0fda08aSwikidesign } else { 12642e80cd5fSwikidesign return 'show'; 1265f0fda08aSwikidesign } 1266f0fda08aSwikidesign } 1267f0fda08aSwikidesign 1268e7c760b3Swikidesign /** 126961437513Swikidesign * Adapted version of pageTemplate() function 127061437513Swikidesign */ 127161437513Swikidesign function _pageTemplate($data) { 127261437513Swikidesign global $conf, $INFO; 127361437513Swikidesign 127461437513Swikidesign $id = $data['id']; 127561437513Swikidesign $user = $_SERVER['REMOTE_USER']; 127661437513Swikidesign $tpl = io_readFile(DOKU_PLUGIN.'discussion/_template.txt'); 127761437513Swikidesign 127861437513Swikidesign // standard replacements 127961437513Swikidesign $replace = array( 128061437513Swikidesign '@NS@' => $data['ns'], 128161437513Swikidesign '@PAGE@' => strtr(noNS($id),'_',' '), 128261437513Swikidesign '@USER@' => $user, 128361437513Swikidesign '@NAME@' => $INFO['userinfo']['name'], 128461437513Swikidesign '@MAIL@' => $INFO['userinfo']['mail'], 1285f014bc86SMichael Klier '@DATE@' => strftime($conf['dformat']), 128661437513Swikidesign ); 128761437513Swikidesign 128861437513Swikidesign // additional replacements 128961437513Swikidesign $replace['@BACK@'] = $data['back']; 129061437513Swikidesign $replace['@TITLE@'] = $data['title']; 129161437513Swikidesign 129261437513Swikidesign // avatar if useavatar and avatar plugin available 129361437513Swikidesign if ($this->getConf('useavatar') 129461437513Swikidesign && (@file_exists(DOKU_PLUGIN.'avatar/syntax.php')) 129561437513Swikidesign && (!plugin_isdisabled('avatar'))) { 129661437513Swikidesign $replace['@AVATAR@'] = '{{avatar>'.$user.' }} '; 129761437513Swikidesign } else { 129861437513Swikidesign $replace['@AVATAR@'] = ''; 129961437513Swikidesign } 130061437513Swikidesign 130161437513Swikidesign // tag if tag plugin is available 130261437513Swikidesign if ((@file_exists(DOKU_PLUGIN.'tag/syntax/tag.php')) 130361437513Swikidesign && (!plugin_isdisabled('tag'))) { 130461437513Swikidesign $replace['@TAG@'] = "\n\n{{tag>}}"; 130561437513Swikidesign } else { 130661437513Swikidesign $replace['@TAG@'] = ''; 130761437513Swikidesign } 130861437513Swikidesign 130961437513Swikidesign // do the replace 131061437513Swikidesign $tpl = str_replace(array_keys($replace), array_values($replace), $tpl); 131161437513Swikidesign return $tpl; 131261437513Swikidesign } 131361437513Swikidesign 131461437513Swikidesign /** 1315e7c760b3Swikidesign * Checks if the CAPTCHA string submitted is valid 1316e7c760b3Swikidesign * 1317e7c760b3Swikidesign * @author Andreas Gohr <gohr@cosmocode.de> 1318e7c760b3Swikidesign * @adaption Esther Brunner <wikidesign@gmail.com> 1319e7c760b3Swikidesign */ 13206046f25cSwikidesign function _captchaCheck() { 1321be1d1f2fSGina Haeussge if (plugin_isdisabled('captcha') || (!$captcha = plugin_load('helper', 'captcha'))) 1322be1d1f2fSGina Haeussge return; // CAPTCHA is disabled or not available 1323e7c760b3Swikidesign 1324d1c29589Swikidesign // do nothing if logged in user and no CAPTCHA required 1325d1c29589Swikidesign if (!$captcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return; 1326d1c29589Swikidesign 1327e7c760b3Swikidesign // compare provided string with decrypted captcha 1328e7c760b3Swikidesign $rand = PMA_blowfish_decrypt($_REQUEST['plugin__captcha_secret'], auth_cookiesalt()); 1329e7c760b3Swikidesign $code = $captcha->_generateCAPTCHA($captcha->_fixedIdent(), $rand); 1330e7c760b3Swikidesign 1331e7c760b3Swikidesign if (!$_REQUEST['plugin__captcha_secret'] || 1332e7c760b3Swikidesign !$_REQUEST['plugin__captcha'] || 1333e7c760b3Swikidesign strtoupper($_REQUEST['plugin__captcha']) != $code) { 1334e7c760b3Swikidesign 1335e7c760b3Swikidesign // CAPTCHA test failed! Continue to edit instead of saving 1336e7c760b3Swikidesign msg($captcha->getLang('testfailed'), -1); 1337e7c760b3Swikidesign if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit'; 1338e7c760b3Swikidesign elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show'; 1339e7c760b3Swikidesign } 1340e7c760b3Swikidesign // if we arrive here it was a valid save 1341e7c760b3Swikidesign } 1342e7c760b3Swikidesign 1343a1ca9e44Swikidesign /** 1344*bd6dc08eSAdrian Schlegel * checks if the submitted reCAPTCHA string is valid 1345*bd6dc08eSAdrian Schlegel * 1346*bd6dc08eSAdrian Schlegel * @author Adrian Schlegel <adrian@liip.ch> 1347*bd6dc08eSAdrian Schlegel */ 1348*bd6dc08eSAdrian Schlegel function _recaptchaCheck() { 1349*bd6dc08eSAdrian Schlegel if (plugin_isdisabled('recaptcha') || (!$recaptcha = plugin_load('helper', 'recaptcha'))) 1350*bd6dc08eSAdrian Schlegel return; // reCAPTCHA is disabled or not available 1351*bd6dc08eSAdrian Schlegel 1352*bd6dc08eSAdrian Schlegel // do nothing if logged in user and no reCAPTCHA required 1353*bd6dc08eSAdrian Schlegel if (!$recaptcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return; 1354*bd6dc08eSAdrian Schlegel 1355*bd6dc08eSAdrian Schlegel $resp = $recaptcha->check(); 1356*bd6dc08eSAdrian Schlegel if (!$resp->is_valid) { 1357*bd6dc08eSAdrian Schlegel msg($recaptcha->getLang('testfailed'),-1); 1358*bd6dc08eSAdrian Schlegel if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit'; 1359*bd6dc08eSAdrian Schlegel elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show'; 1360*bd6dc08eSAdrian Schlegel } 1361*bd6dc08eSAdrian Schlegel } 1362*bd6dc08eSAdrian Schlegel 1363*bd6dc08eSAdrian Schlegel /** 1364a1ca9e44Swikidesign * Adds the comments to the index 1365a1ca9e44Swikidesign */ 136661437513Swikidesign function idx_add_discussion(&$event, $param) { 1367a1ca9e44Swikidesign 1368a1ca9e44Swikidesign // get .comments meta file name 1369a1ca9e44Swikidesign $file = metaFN($event->data[0], '.comments'); 1370a1ca9e44Swikidesign 1371a1ca9e44Swikidesign if (@file_exists($file)) $data = unserialize(io_readFile($file, false)); 1372a1ca9e44Swikidesign if ((!$data['status']) || ($data['number'] == 0)) return; // comments are turned off 1373a1ca9e44Swikidesign 1374a1ca9e44Swikidesign // now add the comments 1375a1ca9e44Swikidesign if (isset($data['comments'])) { 1376a1ca9e44Swikidesign foreach ($data['comments'] as $key => $value) { 137777a22ba2Swikidesign $event->data[1] .= $this->_addCommentWords($key, $data); 1378a1ca9e44Swikidesign } 1379a1ca9e44Swikidesign } 1380a1ca9e44Swikidesign } 1381a1ca9e44Swikidesign 1382a1ca9e44Swikidesign /** 1383a1ca9e44Swikidesign * Adds the words of a given comment to the index 1384a1ca9e44Swikidesign */ 1385efbe59d0Swikidesign function _addCommentWords($cid, &$data, $parent = '') { 1386a1ca9e44Swikidesign 1387efbe59d0Swikidesign if (!isset($data['comments'][$cid])) return ''; // comment was removed 1388a1ca9e44Swikidesign $comment = $data['comments'][$cid]; 1389a1ca9e44Swikidesign 1390efbe59d0Swikidesign if (!is_array($comment)) return ''; // corrupt datatype 1391efbe59d0Swikidesign if ($comment['parent'] != $parent) return ''; // reply to an other comment 1392efbe59d0Swikidesign if (!$comment['show']) return ''; // hidden comment 1393a1ca9e44Swikidesign 1394efbe59d0Swikidesign $text = $comment['raw']; // we only add the raw comment text 1395efbe59d0Swikidesign if (is_array($comment['replies'])) { // and the replies 1396efbe59d0Swikidesign foreach ($comment['replies'] as $rid) { 1397efbe59d0Swikidesign $text .= $this->_addCommentWords($rid, $data, $cid); 1398a1ca9e44Swikidesign } 1399a1ca9e44Swikidesign } 1400efbe59d0Swikidesign return ' '.$text; 1401efbe59d0Swikidesign } 1402a10b5c98SMichael Klier 1403a10b5c98SMichael Klier /** 1404a10b5c98SMichael Klier * Only allow http(s) URLs and append http:// to URLs if needed 1405a10b5c98SMichael Klier */ 1406a10b5c98SMichael Klier function _checkURL($url) { 1407a10b5c98SMichael Klier if(preg_match("#^http://|^https://#", $url)) { 1408a10b5c98SMichael Klier return hsc($url); 1409a10b5c98SMichael Klier } elseif(substr($url, 0, 4) == 'www.') { 1410a10b5c98SMichael Klier return hsc('http://' . $url); 1411a10b5c98SMichael Klier } else { 1412a10b5c98SMichael Klier return ''; 1413a10b5c98SMichael Klier } 1414a10b5c98SMichael Klier } 1415f0fda08aSwikidesign} 141631aab30eSGina Haeussge 141731aab30eSGina Haeussgefunction _sortCallback($a, $b) { 141831aab30eSGina Haeussge if (is_array($a['date'])) { // new format 141931aab30eSGina Haeussge $createdA = $a['date']['created']; 142031aab30eSGina Haeussge } else { // old format 142131aab30eSGina Haeussge $createdA = $a['date']; 142231aab30eSGina Haeussge } 142331aab30eSGina Haeussge 142431aab30eSGina Haeussge if (is_array($b['date'])) { // new format 142531aab30eSGina Haeussge $createdB = $b['date']['created']; 142631aab30eSGina Haeussge } else { // old format 142731aab30eSGina Haeussge $createdB = $b['date']; 142831aab30eSGina Haeussge } 142931aab30eSGina Haeussge 143031aab30eSGina Haeussge if ($createdA == $createdB) 143131aab30eSGina Haeussge return 0; 143231aab30eSGina Haeussge else 143331aab30eSGina Haeussge return ($createdA < $createdB) ? -1 : 1; 143431aab30eSGina Haeussge} 143531aab30eSGina Haeussge 1436530693fbSMichael Klier// vim:ts=4:sw=4:et:enc=utf-8: 1437