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; 135310210d6SGina Haeussge if($btn['type'] == 'linkwiz') continue; 1362d4bee9aSMichael Klier if(preg_match("/=+?/", $btn['open'])) continue; 1372d4bee9aSMichael Klier array_push($toolbar, $btn); 1382d4bee9aSMichael Klier } 1392d4bee9aSMichael Klier $event->data = $toolbar; 1402d4bee9aSMichael Klier } 1412d4bee9aSMichael Klier } 1422d4bee9aSMichael Klier 1432d4bee9aSMichael Klier /** 1442d4bee9aSMichael Klier * Dirty workaround to add a toolbar to the discussion plugin 1452d4bee9aSMichael Klier * 1462d4bee9aSMichael Klier * @author Michael Klier <chi@chimeric.de> 1472d4bee9aSMichael Klier */ 1482d4bee9aSMichael Klier function handle_tpl_metaheader_output(&$event, $param) { 1492d4bee9aSMichael Klier global $ACT; 1502d4bee9aSMichael Klier global $ID; 1512d4bee9aSMichael Klier if($ACT != 'show') return; 1522d4bee9aSMichael Klier 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 @require_once(DOKU_INC.'inc/toolbar.php'); 1562d4bee9aSMichael Klier ob_start(); 1572d4bee9aSMichael Klier print 'NS = "' . getNS($ID) . '";'; // we have to define NS, otherwise we get get JS errors 1582d4bee9aSMichael Klier toolbar_JSdefines('toolbar'); 1592d4bee9aSMichael Klier $script = ob_get_clean(); 1602d4bee9aSMichael Klier array_push($event->data['script'], array('type' => 'text/javascript', 'charset' => "utf-8", '_data' => $script)); 1612d4bee9aSMichael Klier } 162f0fda08aSwikidesign } 163f0fda08aSwikidesign 164f0fda08aSwikidesign /** 165a1d93126SGina Haeussge * Handles comment actions, dispatches data processing routines 166f0fda08aSwikidesign */ 167a1d93126SGina Haeussge function handle_act_preprocess(&$event, $param) { 168284f9aa2SMichael Klier global $ID; 169d7eb2e24SMichael Klier global $INFO; 170284f9aa2SMichael Klier global $conf; 1713011fb8bSMichael Klier global $lang; 172573e23a1Swikidesign 173a1d93126SGina Haeussge // handle newthread ACTs 174a1d93126SGina Haeussge if ($event->data == 'newthread') { 175a1d93126SGina Haeussge // we can handle it -> prevent others 176a1d93126SGina Haeussge $event->preventDefault(); 177a1d93126SGina Haeussge $event->data = $this->_newThread(); 178a1d93126SGina Haeussge } 179a1d93126SGina Haeussge 180a1d93126SGina Haeussge // enable captchas 181bd6dc08eSAdrian Schlegel if (in_array($_REQUEST['comment'], array('add', 'save'))) { 182bd6dc08eSAdrian Schlegel if (@file_exists(DOKU_PLUGIN.'captcha/action.php')) { 183a1d93126SGina Haeussge $this->_captchaCheck(); 184a1d93126SGina Haeussge } 185bd6dc08eSAdrian Schlegel if (@file_exists(DOKU_PLUGIN.'recaptcha/action.php')) { 186bd6dc08eSAdrian Schlegel $this->_recaptchaCheck(); 187bd6dc08eSAdrian Schlegel } 188bd6dc08eSAdrian Schlegel } 189a1d93126SGina Haeussge 1903011fb8bSMichael Klier // if we are not in show mode or someone wants to unsubscribe, that was all for now 191b2f2e866SMichael Klier if ($event->data != 'show' && $event->data != 'discussion_unsubscribe' && $event->data != 'discussion_confirmsubscribe') return; 192a1d93126SGina Haeussge 193b2f2e866SMichael Klier if ($event->data == 'discussion_unsubscribe' or $event->data == 'discussion_confirmsubscribe') { 1949881d835SMichael Klier // ok we can handle it prevent others 1959881d835SMichael Klier $event->preventDefault(); 1969881d835SMichael Klier 1973011fb8bSMichael Klier if (!isset($_REQUEST['hash'])) { 1989881d835SMichael Klier return false; 1993011fb8bSMichael Klier } else { 2003011fb8bSMichael Klier $file = metaFN($ID, '.comments'); 2013011fb8bSMichael Klier $data = unserialize(io_readFile($file)); 2020c54624fSGina Haeussge $themail = ''; 2039881d835SMichael Klier foreach($data['subscribers'] as $mail => $info) { 2049881d835SMichael Klier // convert old style subscribers just in case 2059881d835SMichael Klier if(!is_array($info)) { 2069881d835SMichael Klier $hash = $data['subscribers'][$mail]; 2079881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 2089881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 2099881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 2109881d835SMichael Klier } 2119881d835SMichael Klier 2129881d835SMichael Klier if ($data['subscribers'][$mail]['hash'] == $_REQUEST['hash']) { 2130c54624fSGina Haeussge $themail = $mail; 2140c54624fSGina Haeussge } 2150c54624fSGina Haeussge } 2160c54624fSGina Haeussge 2170c54624fSGina Haeussge if($themail != '') { 218b2f2e866SMichael Klier if($event->data == 'discussion_unsubscribe') { 2190c54624fSGina Haeussge unset($data['subscribers'][$themail]); 2200c54624fSGina Haeussge msg(sprintf($lang['unsubscribe_success'], $themail, $ID), 1); 221b2f2e866SMichael Klier } elseif($event->data == 'discussion_confirmsubscribe') { 2220c54624fSGina Haeussge $data['subscribers'][$themail]['active'] = true; 2230c54624fSGina Haeussge msg(sprintf($lang['subscribe_success'], $themail, $ID), 1); 2249881d835SMichael Klier } 2259881d835SMichael Klier io_saveFile($file, serialize($data)); 2263011fb8bSMichael Klier $event->data = 'show'; 2273011fb8bSMichael Klier return true; 2289881d835SMichael Klier } else { 2293011fb8bSMichael Klier return false; 2303011fb8bSMichael Klier } 2319881d835SMichael Klier } 2323011fb8bSMichael Klier } else { 233a1d93126SGina Haeussge // do the data processing for comments 234f0fda08aSwikidesign $cid = $_REQUEST['cid']; 235f0fda08aSwikidesign switch ($_REQUEST['comment']) { 236f0fda08aSwikidesign case 'add': 23737e3c825SMichael Klier if(empty($_REQUEST['text'])) return; // don't add empty comments 23894c5d164SMichael Klier if(isset($_SERVER['REMOTE_USER']) && !$this->getConf('adminimport')) { 23994c5d164SMichael Klier $comment['user']['id'] = $_SERVER['REMOTE_USER']; 24094c5d164SMichael Klier $comment['user']['name'] = $INFO['userinfo']['name']; 24194c5d164SMichael Klier $comment['user']['mail'] = $INFO['userinfo']['mail']; 24294c5d164SMichael Klier } elseif((isset($_SERVER['REMOTE_USER']) && $this->getConf('adminimport') && auth_ismanager()) || !isset($_SERVER['REMOTE_USER'])) { 243*c9d36b5eSMichael Klier if(empty($_REQUEST['name']) or empty($_REQUEST['mail'])) return; // don't add anonymous comments 244*c9d36b5eSMichael Klier if(!mail_isvalid($_REQUEST['mail'])) { 245*c9d36b5eSMichael Klier msg($lang['regbadmail'], -1); 246*c9d36b5eSMichael Klier return; 247*c9d36b5eSMichael Klier } else { 24894c5d164SMichael Klier $comment['user']['id'] = 'test'.hsc($_REQUEST['user']); 24994c5d164SMichael Klier $comment['user']['name'] = hsc($_REQUEST['name']); 25094c5d164SMichael Klier $comment['user']['mail'] = hsc($_REQUEST['mail']); 25194c5d164SMichael Klier } 252*c9d36b5eSMichael Klier } 25394c5d164SMichael Klier $comment['user']['address'] = ($this->getConf('addressfield')) ? hsc($_REQUEST['address']) : ''; 254a10b5c98SMichael Klier $comment['user']['url'] = ($this->getConf('urlfield')) ? $this->_checkURL($_REQUEST['url']) : ''; 25594c5d164SMichael Klier $comment['subscribe'] = ($this->getConf('subscribe')) ? $_REQUEST['subscribe'] : ''; 25694c5d164SMichael Klier $comment['date'] = array('created' => $_REQUEST['date']); 25794c5d164SMichael Klier $comment['raw'] = cleanText($_REQUEST['text']); 258f0fda08aSwikidesign $repl = $_REQUEST['reply']; 259b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 260a44bc9f7SMichael Klier $comment['show'] = false; 261a44bc9f7SMichael Klier } else { 262a44bc9f7SMichael Klier $comment['show'] = true; 263a44bc9f7SMichael Klier } 264f0fda08aSwikidesign $this->_add($comment, $repl); 265f0fda08aSwikidesign break; 266f0fda08aSwikidesign 267f0fda08aSwikidesign case 'save': 268f0fda08aSwikidesign $raw = cleanText($_REQUEST['text']); 269264b7327Swikidesign $this->_save(array($cid), $raw); 270f0fda08aSwikidesign break; 271f0fda08aSwikidesign 2721e46d176Swikidesign case 'delete': 273264b7327Swikidesign $this->_save(array($cid), ''); 2742ee3dca3Swikidesign break; 2751e46d176Swikidesign 276f0fda08aSwikidesign case 'toogle': 277264b7327Swikidesign $this->_save(array($cid), '', 'toogle'); 278f0fda08aSwikidesign break; 279a1d93126SGina Haeussge } 2803011fb8bSMichael Klier } 281a1d93126SGina Haeussge } 282a1d93126SGina Haeussge 283a1d93126SGina Haeussge /** 284a1d93126SGina Haeussge * Main function; dispatches the visual comment actions 285a1d93126SGina Haeussge */ 286a1d93126SGina Haeussge function comments(&$event, $param) { 287a1d93126SGina Haeussge if ($event->data != 'show') return; // nothing to do for us 288a1d93126SGina Haeussge 289a1d93126SGina Haeussge $cid = $_REQUEST['cid']; 290a1d93126SGina Haeussge switch ($_REQUEST['comment']) { 291a1d93126SGina Haeussge case 'edit': 292a1d93126SGina Haeussge $this->_show(NULL, $cid); 293a1d93126SGina Haeussge break; 2942b18adb9SMichael Klier default: 295f0fda08aSwikidesign $this->_show($cid); 2962b18adb9SMichael Klier break; 297f0fda08aSwikidesign } 298f0fda08aSwikidesign } 299f0fda08aSwikidesign 300f0fda08aSwikidesign /** 301a1d93126SGina Haeussge * Redirects browser to given comment anchor 302a1d93126SGina Haeussge */ 303a1d93126SGina Haeussge function _redirect($cid) { 304a1d93126SGina Haeussge global $ID; 305a1d93126SGina Haeussge global $ACT; 306a1d93126SGina Haeussge 307a1d93126SGina Haeussge if ($ACT !== 'show') return; 308a44bc9f7SMichael Klier 309b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 310a44bc9f7SMichael Klier msg($this->getLang('moderation'), 1); 311a44bc9f7SMichael Klier @session_start(); 312a44bc9f7SMichael Klier global $MSG; 313a44bc9f7SMichael Klier $_SESSION[DOKU_COOKIE]['msg'] = $MSG; 314a44bc9f7SMichael Klier session_write_close(); 315a44bc9f7SMichael Klier $url = wl($ID); 316a44bc9f7SMichael Klier } else { 317a44bc9f7SMichael Klier $url = wl($ID) . '#comment_' . $cid; 318a44bc9f7SMichael Klier } 319a44bc9f7SMichael Klier send_redirect($url); 3206d1f4f20SGina Haeussge exit(); 321a1d93126SGina Haeussge } 322a1d93126SGina Haeussge 323a1d93126SGina Haeussge /** 324f0fda08aSwikidesign * Shows all comments of the current page 325f0fda08aSwikidesign */ 326f0fda08aSwikidesign function _show($reply = NULL, $edit = NULL) { 3272b18adb9SMichael Klier global $ID; 3282b18adb9SMichael Klier global $INFO; 3292b18adb9SMichael Klier global $ACT; 330573e23a1Swikidesign 331479dd10fSwikidesign // get .comments meta file name 332f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 333f0fda08aSwikidesign 3344aeddd8eSMichael Klier if (!$INFO['exists']) return; 3352b18adb9SMichael Klier if (!@file_exists($file) && !$this->getConf('automatic')) return false; 3367d091ad1SMichael Klier if (!$_SERVER['REMOTE_USER'] && !$this->getConf('showguests')) return false; 337f0fda08aSwikidesign 3382b18adb9SMichael Klier // load data 3392b18adb9SMichael Klier if (@file_exists($file)) { 3402b18adb9SMichael Klier $data = unserialize(io_readFile($file, false)); 341479dd10fSwikidesign if (!$data['status']) return false; // comments are turned off 3422b18adb9SMichael Klier } elseif (!@file_exists($file) && $this->getConf('automatic') && $INFO['exists']) { 3432b18adb9SMichael Klier // set status to show the comment form 3442b18adb9SMichael Klier $data['status'] = 1; 3452b18adb9SMichael Klier $data['number'] = 0; 3462b18adb9SMichael Klier } 347f0fda08aSwikidesign 348a1599850SMichael Klier // show discussion wrapper only on certain circumstances 349a1599850SMichael Klier $cnt = count($data['comments']); 35064018b99SMichael Klier $keys = @array_keys($data['comments']); 351a1599850SMichael Klier if($cnt > 1 || ($cnt == 1 && $data['comments'][$keys[0]]['show'] == 1) || $this->getConf('allowguests') || isset($_SERVER['REMOTE_USER'])) { 352a1599850SMichael Klier $show = true; 353f0fda08aSwikidesign // section title 35407c376bbSwikidesign $title = ($data['title'] ? hsc($data['title']) : $this->getLang('discussion')); 3554a0a1bd2Swikidesign ptln('<div class="comment_wrapper">'); 3564a0a1bd2Swikidesign ptln('<h2><a name="discussion__section" id="discussion__section">', 2); 3574a0a1bd2Swikidesign ptln($title, 4); 3584a0a1bd2Swikidesign ptln('</a></h2>', 2); 3594a0a1bd2Swikidesign ptln('<div class="level2 hfeed">', 2); 360a1599850SMichael Klier } 361a1599850SMichael Klier 362f0fda08aSwikidesign // now display the comments 363f0fda08aSwikidesign if (isset($data['comments'])) { 36431aab30eSGina Haeussge if (!$this->getConf('usethreading')) { 36531aab30eSGina Haeussge $data['comments'] = $this->_flattenThreads($data['comments']); 36631aab30eSGina Haeussge uasort($data['comments'], '_sortCallBack'); 36731aab30eSGina Haeussge } 368dbd9d5cdSMichael Klier if($this->getConf('newestfirst')) { 369dbd9d5cdSMichael Klier $data['comments'] = array_reverse($data['comments']); 370dbd9d5cdSMichael Klier } 371f0fda08aSwikidesign foreach ($data['comments'] as $key => $value) { 372f0fda08aSwikidesign if ($key == $edit) $this->_form($value['raw'], 'save', $edit); // edit form 373f0fda08aSwikidesign else $this->_print($key, $data, '', $reply); 374f0fda08aSwikidesign } 375f0fda08aSwikidesign } 376f0fda08aSwikidesign 377f0fda08aSwikidesign // comment form 37831aab30eSGina Haeussge if (($data['status'] == 1) && (!$reply || !$this->getConf('usethreading')) && !$edit) $this->_form(''); 379f0fda08aSwikidesign 380a1599850SMichael Klier if($show) { 3814a0a1bd2Swikidesign ptln('</div>', 2); // level2 hfeed 3824a0a1bd2Swikidesign ptln('</div>'); // comment_wrapper 383a1599850SMichael Klier } 384f0fda08aSwikidesign 385f0fda08aSwikidesign return true; 386f0fda08aSwikidesign } 387f0fda08aSwikidesign 38831aab30eSGina Haeussge function _flattenThreads($comments, $keys = null) { 38931aab30eSGina Haeussge if (is_null($keys)) 39031aab30eSGina Haeussge $keys = array_keys($comments); 39131aab30eSGina Haeussge 39231aab30eSGina Haeussge foreach($keys as $cid) { 39331aab30eSGina Haeussge if (!empty($comments[$cid]['replies'])) { 39431aab30eSGina Haeussge $rids = $comments[$cid]['replies']; 39531aab30eSGina Haeussge $comments = $this->_flattenThreads($comments, $rids); 39631aab30eSGina Haeussge $comments[$cid]['replies'] = array(); 39731aab30eSGina Haeussge } 39831aab30eSGina Haeussge $comments[$cid]['parent'] = ''; 39931aab30eSGina Haeussge } 40031aab30eSGina Haeussge return $comments; 40131aab30eSGina Haeussge } 40231aab30eSGina Haeussge 403f0fda08aSwikidesign /** 404f0fda08aSwikidesign * Adds a new comment and then displays all comments 405f0fda08aSwikidesign */ 406f0fda08aSwikidesign function _add($comment, $parent) { 4073011fb8bSMichael Klier global $lang; 4083011fb8bSMichael Klier global $ID; 4093011fb8bSMichael Klier global $TEXT; 410f0fda08aSwikidesign 411f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 412f0fda08aSwikidesign $TEXT = $comment['raw']; 413f0fda08aSwikidesign 414f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 415f0fda08aSwikidesign if (checkwordblock()) { 416f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 417f0fda08aSwikidesign return false; 418f0fda08aSwikidesign } 419f0fda08aSwikidesign 4203011fb8bSMichael Klier if ((!$this->getConf('allowguests')) 4213011fb8bSMichael Klier && ($comment['user']['id'] != $_SERVER['REMOTE_USER'])) 4223011fb8bSMichael Klier return false; // guest comments not allowed 4233011fb8bSMichael Klier 424f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 425f0fda08aSwikidesign 426f0fda08aSwikidesign // get discussion meta file name 427f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 428f0fda08aSwikidesign 4292b18adb9SMichael Klier // create comments file if it doesn't exist yet 4302b18adb9SMichael Klier if(!@file_exists($file)) { 4312b18adb9SMichael Klier $data = array('status' => 1, 'number' => 0); 4322b18adb9SMichael Klier io_saveFile($file, serialize($data)); 4332b18adb9SMichael Klier } else { 434f0fda08aSwikidesign $data = array(); 435f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 436f0fda08aSwikidesign if ($data['status'] != 1) return false; // comments off or closed 4372b18adb9SMichael Klier } 4382b18adb9SMichael Klier 4393011fb8bSMichael Klier if ($comment['date']['created']) { 4403011fb8bSMichael Klier $date = strtotime($comment['date']['created']); 4413011fb8bSMichael Klier } else { 4423011fb8bSMichael Klier $date = time(); 4433011fb8bSMichael Klier } 444f0fda08aSwikidesign 4453011fb8bSMichael Klier if ($date == -1) { 4463011fb8bSMichael Klier $date = time(); 4473011fb8bSMichael Klier } 4483011fb8bSMichael Klier 4496046f25cSwikidesign $cid = md5($comment['user']['id'].$date); // create a unique id 450f0fda08aSwikidesign 4513011fb8bSMichael Klier if (!is_array($data['comments'][$parent])) { 4523011fb8bSMichael Klier $parent = NULL; // invalid parent comment 4533011fb8bSMichael Klier } 454f0fda08aSwikidesign 455f0fda08aSwikidesign // render the comment 456f0fda08aSwikidesign $xhtml = $this->_render($comment['raw']); 457f0fda08aSwikidesign 458f0fda08aSwikidesign // fill in the new comment 459f0fda08aSwikidesign $data['comments'][$cid] = array( 4606046f25cSwikidesign 'user' => $comment['user'], 4616046f25cSwikidesign 'date' => array('created' => $date), 462f0fda08aSwikidesign 'show' => true, 4636046f25cSwikidesign 'raw' => $comment['raw'], 464f0fda08aSwikidesign 'xhtml' => $xhtml, 465f0fda08aSwikidesign 'parent' => $parent, 466a44bc9f7SMichael Klier 'replies' => array(), 467a44bc9f7SMichael Klier 'show' => $comment['show'] 468f0fda08aSwikidesign ); 469f0fda08aSwikidesign 4703011fb8bSMichael Klier if($comment['subscribe']) { 4713011fb8bSMichael Klier $mail = $comment['user']['mail']; 4723011fb8bSMichael Klier if($data['subscribers']) { 4733011fb8bSMichael Klier if(!$data['subscribers'][$mail]) { 4749881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4759881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4769881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4779881d835SMichael Klier } else { 4789881d835SMichael Klier // convert old style subscribers and set them active 4799881d835SMichael Klier if(!is_array($data['subscribers'][$mail])) { 4809881d835SMichael Klier $hash = $data['subscribers'][$mail]; 4819881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 4829881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 4839881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 4849881d835SMichael Klier } 4853011fb8bSMichael Klier } 4863011fb8bSMichael Klier } else { 4879881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4889881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4899881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4903011fb8bSMichael Klier } 4913011fb8bSMichael Klier } 4923011fb8bSMichael Klier 493f0fda08aSwikidesign // update parent comment 494f0fda08aSwikidesign if ($parent) $data['comments'][$parent]['replies'][] = $cid; 495f0fda08aSwikidesign 496f0fda08aSwikidesign // update the number of comments 497f0fda08aSwikidesign $data['number']++; 498f0fda08aSwikidesign 499f0fda08aSwikidesign // notify subscribers of the page 5008b42cefbSMichael Klier $data['comments'][$cid]['cid'] = $cid; 5013011fb8bSMichael Klier $this->_notify($data['comments'][$cid], $data['subscribers']); 502f0fda08aSwikidesign 5039881d835SMichael Klier // save the comment metadata file 5049881d835SMichael Klier io_saveFile($file, serialize($data)); 5059881d835SMichael Klier $this->_addLogEntry($date, $ID, 'cc', '', $cid); 5069881d835SMichael Klier 507a1d93126SGina Haeussge $this->_redirect($cid); 508f0fda08aSwikidesign return true; 509f0fda08aSwikidesign } 510f0fda08aSwikidesign 511f0fda08aSwikidesign /** 512f0fda08aSwikidesign * Saves the comment with the given ID and then displays all comments 513f0fda08aSwikidesign */ 514264b7327Swikidesign function _save($cids, $raw, $act = NULL) { 515a1ace3c9Swikidesign global $ID; 516f0fda08aSwikidesign 517757550e8SMichael Klier if(!$cids) return; // do nothing if we get no comment id 518757550e8SMichael Klier 5192ee3dca3Swikidesign if ($raw) { 5202ee3dca3Swikidesign global $TEXT; 5212ee3dca3Swikidesign 522f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 523f0fda08aSwikidesign $TEXT = $raw; 524f0fda08aSwikidesign 525f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 526f0fda08aSwikidesign if (checkwordblock()) { 527f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 528f0fda08aSwikidesign return false; 529f0fda08aSwikidesign } 530f0fda08aSwikidesign 531f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 5322ee3dca3Swikidesign } 533f0fda08aSwikidesign 534f0fda08aSwikidesign // get discussion meta file name 535f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 536f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 537f0fda08aSwikidesign 538573e23a1Swikidesign if (!is_array($cids)) $cids = array($cids); 539264b7327Swikidesign foreach ($cids as $cid) { 540264b7327Swikidesign 5416046f25cSwikidesign if (is_array($data['comments'][$cid]['user'])) { 5426046f25cSwikidesign $user = $data['comments'][$cid]['user']['id']; 5436046f25cSwikidesign $convert = false; 5446046f25cSwikidesign } else { 5456046f25cSwikidesign $user = $data['comments'][$cid]['user']; 5466046f25cSwikidesign $convert = true; 5476046f25cSwikidesign } 5486046f25cSwikidesign 549f0fda08aSwikidesign // someone else was trying to edit our comment -> abort 5503e02b3ffSwikidesign if (($user != $_SERVER['REMOTE_USER']) && (!auth_ismanager())) return false; 551f0fda08aSwikidesign 552f0fda08aSwikidesign $date = time(); 553f0fda08aSwikidesign 5546046f25cSwikidesign // need to convert to new format? 5556046f25cSwikidesign if ($convert) { 5566046f25cSwikidesign $data['comments'][$cid]['user'] = array( 5576046f25cSwikidesign 'id' => $user, 5586046f25cSwikidesign 'name' => $data['comments'][$cid]['name'], 5596046f25cSwikidesign 'mail' => $data['comments'][$cid]['mail'], 5606046f25cSwikidesign 'url' => $data['comments'][$cid]['url'], 5616046f25cSwikidesign 'address' => $data['comments'][$cid]['address'], 5626046f25cSwikidesign ); 5636046f25cSwikidesign $data['comments'][$cid]['date'] = array( 5646046f25cSwikidesign 'created' => $data['comments'][$cid]['date'] 5656046f25cSwikidesign ); 5666046f25cSwikidesign } 5676046f25cSwikidesign 568264b7327Swikidesign if ($act == 'toogle') { // toogle visibility 569f0fda08aSwikidesign $now = $data['comments'][$cid]['show']; 570f0fda08aSwikidesign $data['comments'][$cid]['show'] = !$now; 571f0fda08aSwikidesign $data['number'] = $this->_count($data); 572f0fda08aSwikidesign 573f0fda08aSwikidesign $type = ($data['comments'][$cid]['show'] ? 'sc' : 'hc'); 574f0fda08aSwikidesign 575264b7327Swikidesign } elseif ($act == 'show') { // show comment 576264b7327Swikidesign $data['comments'][$cid]['show'] = true; 577264b7327Swikidesign $data['number'] = $this->_count($data); 578264b7327Swikidesign 579573e23a1Swikidesign $type = 'sc'; // show comment 580264b7327Swikidesign 581264b7327Swikidesign } elseif ($act == 'hide') { // hide comment 582264b7327Swikidesign $data['comments'][$cid]['show'] = false; 583264b7327Swikidesign $data['number'] = $this->_count($data); 584264b7327Swikidesign 585573e23a1Swikidesign $type = 'hc'; // hide comment 586264b7327Swikidesign 587f0fda08aSwikidesign } elseif (!$raw) { // remove the comment 588efbe59d0Swikidesign $data['comments'] = $this->_removeComment($cid, $data['comments']); 589f0fda08aSwikidesign $data['number'] = $this->_count($data); 590f0fda08aSwikidesign 591573e23a1Swikidesign $type = 'dc'; // delete comment 592f0fda08aSwikidesign 593f0fda08aSwikidesign } else { // save changed comment 594f0fda08aSwikidesign $xhtml = $this->_render($raw); 595f0fda08aSwikidesign 596f0fda08aSwikidesign // now change the comment's content 5976046f25cSwikidesign $data['comments'][$cid]['date']['modified'] = $date; 5986046f25cSwikidesign $data['comments'][$cid]['raw'] = $raw; 599f0fda08aSwikidesign $data['comments'][$cid]['xhtml'] = $xhtml; 600f0fda08aSwikidesign 601573e23a1Swikidesign $type = 'ec'; // edit comment 602f0fda08aSwikidesign } 603264b7327Swikidesign } 604264b7327Swikidesign 605f0fda08aSwikidesign // save the comment metadata file 606f0fda08aSwikidesign io_saveFile($file, serialize($data)); 607f0fda08aSwikidesign $this->_addLogEntry($date, $ID, $type, '', $cid); 608f0fda08aSwikidesign 609a1d93126SGina Haeussge $this->_redirect($cid); 610f0fda08aSwikidesign return true; 611f0fda08aSwikidesign } 612f0fda08aSwikidesign 613f0fda08aSwikidesign /** 614efbe59d0Swikidesign * Recursive function to remove a comment 615efbe59d0Swikidesign */ 616efbe59d0Swikidesign function _removeComment($cid, $comments) { 617efbe59d0Swikidesign if (is_array($comments[$cid]['replies'])) { 618efbe59d0Swikidesign foreach ($comments[$cid]['replies'] as $rid) { 619efbe59d0Swikidesign $comments = $this->_removeComment($rid, $comments); 620efbe59d0Swikidesign } 621efbe59d0Swikidesign } 622efbe59d0Swikidesign unset($comments[$cid]); 623efbe59d0Swikidesign return $comments; 624efbe59d0Swikidesign } 625efbe59d0Swikidesign 626efbe59d0Swikidesign /** 627f0fda08aSwikidesign * Prints an individual comment 628f0fda08aSwikidesign */ 629f0fda08aSwikidesign function _print($cid, &$data, $parent = '', $reply = '', $visible = true) { 630f0fda08aSwikidesign 6312ee3dca3Swikidesign if (!isset($data['comments'][$cid])) return false; // comment was removed 632f0fda08aSwikidesign $comment = $data['comments'][$cid]; 633f0fda08aSwikidesign 634f0fda08aSwikidesign if (!is_array($comment)) return false; // corrupt datatype 635f0fda08aSwikidesign 636f0fda08aSwikidesign if ($comment['parent'] != $parent) return true; // reply to an other comment 637f0fda08aSwikidesign 638f0fda08aSwikidesign if (!$comment['show']) { // comment hidden 639a1ace3c9Swikidesign if (auth_ismanager()) $hidden = ' comment_hidden'; 640f0fda08aSwikidesign else return true; 6414a0a1bd2Swikidesign } else { 6424a0a1bd2Swikidesign $hidden = ''; 643f0fda08aSwikidesign } 644f0fda08aSwikidesign 645f1c6610eSpierre.spring // print the actual comment 6466b1c1263SMichael Klier $this->_print_comment($cid, $data, $parent, $reply, $visible, $hidden); 647f1c6610eSpierre.spring // replies to this comment entry? 648f1c6610eSpierre.spring $this->_print_replies($cid, $data, $reply, $visible); 649f1c6610eSpierre.spring // reply form 650f1c6610eSpierre.spring $this->_print_form($cid, $reply); 651f1c6610eSpierre.spring } 652f1c6610eSpierre.spring 653f1c6610eSpierre.spring function _print_comment($cid, &$data, $parent, $reply, $visible, $hidden) 654f1c6610eSpierre.spring { 655f1c6610eSpierre.spring global $conf, $lang, $ID, $HIGH; 656f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 657f1c6610eSpierre.spring 658f0fda08aSwikidesign // comment head with date and user data 6594a0a1bd2Swikidesign ptln('<div class="hentry'.$hidden.'">', 4); 6604a0a1bd2Swikidesign ptln('<div class="comment_head">', 6); 6611810ba9cSMichael Klier ptln('<a name="comment_'.$cid.'" id="comment_'.$cid.'"></a>', 8); 6624a0a1bd2Swikidesign $head = '<span class="vcard author">'; 663f0fda08aSwikidesign 6646046f25cSwikidesign // prepare variables 6656046f25cSwikidesign if (is_array($comment['user'])) { // new format 6666046f25cSwikidesign $user = $comment['user']['id']; 6676046f25cSwikidesign $name = $comment['user']['name']; 6686046f25cSwikidesign $mail = $comment['user']['mail']; 6696046f25cSwikidesign $url = $comment['user']['url']; 6706046f25cSwikidesign $address = $comment['user']['address']; 6716046f25cSwikidesign } else { // old format 6726046f25cSwikidesign $user = $comment['user']; 6736046f25cSwikidesign $name = $comment['name']; 6746046f25cSwikidesign $mail = $comment['mail']; 6756046f25cSwikidesign $url = $comment['url']; 6766046f25cSwikidesign $address = $comment['address']; 6776046f25cSwikidesign } 6786046f25cSwikidesign if (is_array($comment['date'])) { // new format 6796046f25cSwikidesign $created = $comment['date']['created']; 6806046f25cSwikidesign $modified = $comment['date']['modified']; 6816046f25cSwikidesign } else { // old format 6826046f25cSwikidesign $created = $comment['date']; 6836046f25cSwikidesign $modified = $comment['edited']; 6846046f25cSwikidesign } 6856046f25cSwikidesign 6860ff5ab97SMichael Klier // show username or real name? 6870ff5ab97SMichael Klier if ((!$this->getConf('userealname')) && ($user)) { 6880ff5ab97SMichael Klier $showname = $user; 6890ff5ab97SMichael Klier } else { 6900ff5ab97SMichael Klier $showname = $name; 6910ff5ab97SMichael Klier } 6920ff5ab97SMichael Klier 693ce7b17cfSwikidesign // show avatar image? 694f1c6610eSpierre.spring if ($this->_use_avatar()) { 69512ca6034SMichael Klier $user_data['name'] = $name; 69612ca6034SMichael Klier $user_data['user'] = $user; 69712ca6034SMichael Klier $user_data['mail'] = $mail; 69812ca6034SMichael Klier $avatar = $this->avatar->getXHTML($user_data, $name, 'left'); 699db963b87SMichael Klier if($avatar) $head .= $avatar; 700f0fda08aSwikidesign } 701f0fda08aSwikidesign 7026046f25cSwikidesign if ($this->getConf('linkemail') && $mail) { 7030ff5ab97SMichael Klier $head .= $this->email($mail, $showname, 'email fn'); 7046046f25cSwikidesign } elseif ($url) { 7050ff5ab97SMichael Klier $head .= $this->external_link($this->_checkURL($url), $showname, 'urlextern url fn'); 706f0fda08aSwikidesign } else { 7070ff5ab97SMichael Klier $head .= '<span class="fn">'.$showname.'</span>'; 708f0fda08aSwikidesign } 7094a0a1bd2Swikidesign if ($address) $head .= ', <span class="adr">'.$address.'</span>'; 7104a0a1bd2Swikidesign $head .= '</span>, '. 711f014bc86SMichael Klier '<abbr class="published" title="'.strftime('%Y-%m-%dT%H:%M:%SZ', $created).'">'. 712f014bc86SMichael Klier strftime($conf['dformat'], $created).'</abbr>'; 7134a0a1bd2Swikidesign if ($comment['edited']) $head .= ' (<abbr class="updated" title="'. 714f014bc86SMichael Klier strftime('%Y-%m-%dT%H:%M:%SZ', $modified).'">'.strftime($conf['dformat'], $modified). 7156046f25cSwikidesign '</abbr>)'; 716f014bc86SMichael Klier ptln($head, 8); 7174a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_head" 718f0fda08aSwikidesign 719f0fda08aSwikidesign // main comment content 7204a0a1bd2Swikidesign ptln('<div class="comment_body entry-content"'. 721f1c6610eSpierre.spring ($this->getConf('useavatar') ? $this->_get_style() : '').'>', 6); 72215cdad37Spierre.spring echo ($HIGH?html_hilight($comment['xhtml'],$HIGH):$comment['xhtml']).DOKU_LF; 7234a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_body" 724f0fda08aSwikidesign 725f0fda08aSwikidesign if ($visible) { 7261184c36aSwikidesign ptln('<div class="comment_buttons">', 6); 727f0fda08aSwikidesign 728f0fda08aSwikidesign // show reply button? 729f1c4aa1aSwikidesign if (($data['status'] == 1) && !$reply && $comment['show'] 73031aab30eSGina Haeussge && ($this->getConf('allowguests') || $_SERVER['REMOTE_USER']) && $this->getConf('usethreading')) 7311e46d176Swikidesign $this->_button($cid, $this->getLang('btn_reply'), 'reply', true); 732f0fda08aSwikidesign 7331184c36aSwikidesign // show edit, show/hide and delete button? 7341184c36aSwikidesign if ((($user == $_SERVER['REMOTE_USER']) && ($user != '')) || (auth_ismanager())) { 7351e46d176Swikidesign $this->_button($cid, $lang['btn_secedit'], 'edit', true); 7361184c36aSwikidesign $label = ($comment['show'] ? $this->getLang('btn_hide') : $this->getLang('btn_show')); 7371184c36aSwikidesign $this->_button($cid, $label, 'toogle'); 7381184c36aSwikidesign $this->_button($cid, $lang['btn_delete'], 'delete'); 739f0fda08aSwikidesign } 7401184c36aSwikidesign ptln('</div>', 6); // class="comment_buttons" 7411184c36aSwikidesign } 7421184c36aSwikidesign ptln('</div>', 4); // class="hentry" 743f0fda08aSwikidesign } 744f0fda08aSwikidesign 745f1c6610eSpierre.spring function _print_form($cid, $reply) 746f1c6610eSpierre.spring { 74731aab30eSGina Haeussge if ($this->getConf('usethreading') && $reply == $cid) { 7484a0a1bd2Swikidesign ptln('<div class="comment_replies">', 4); 749f0fda08aSwikidesign $this->_form('', 'add', $cid); 7504a0a1bd2Swikidesign ptln('</div>', 4); // class="comment_replies" 751f0fda08aSwikidesign } 752f0fda08aSwikidesign } 753f0fda08aSwikidesign 754f1c6610eSpierre.spring function _print_replies($cid, &$data, $reply, &$visible) 755f1c6610eSpierre.spring { 756f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 757f1c6610eSpierre.spring if (!count($comment['replies'])) { 758f1c6610eSpierre.spring return; 759f1c6610eSpierre.spring } 760f1c6610eSpierre.spring ptln('<div class="comment_replies"'.$this->_get_style().'>', 4); 761f1c6610eSpierre.spring $visible = ($comment['show'] && $visible); 762f1c6610eSpierre.spring foreach ($comment['replies'] as $rid) { 763f1c6610eSpierre.spring $this->_print($rid, $data, $cid, $reply, $visible); 764f1c6610eSpierre.spring } 765f1c6610eSpierre.spring ptln('</div>', 4); 766f1c6610eSpierre.spring } 767f1c6610eSpierre.spring 768f1c6610eSpierre.spring function _use_avatar() 769f1c6610eSpierre.spring { 770f1c6610eSpierre.spring if (is_null($this->use_avatar)) { 771f1c6610eSpierre.spring $this->use_avatar = $this->getConf('useavatar') 772f1c6610eSpierre.spring && (!plugin_isdisabled('avatar')) 773f1c6610eSpierre.spring && ($this->avatar =& plugin_load('helper', 'avatar')); 774f1c6610eSpierre.spring } 775f1c6610eSpierre.spring return $this->use_avatar; 776f1c6610eSpierre.spring } 777f1c6610eSpierre.spring 778f1c6610eSpierre.spring function _get_style() 779f1c6610eSpierre.spring { 780f1c6610eSpierre.spring if (is_null($this->style)){ 781f1c6610eSpierre.spring if ($this->_use_avatar()) { 782f1c6610eSpierre.spring $this->style = ' style="margin-left: '.($this->avatar->getConf('size') + 14).'px;"'; 783f1c6610eSpierre.spring } else { 784f1c6610eSpierre.spring $this->style = ' style="margin-left: 20px;"'; 785f1c6610eSpierre.spring } 786f1c6610eSpierre.spring } 787f1c6610eSpierre.spring return $this->style; 788f1c6610eSpierre.spring } 789f1c6610eSpierre.spring 790f0fda08aSwikidesign /** 791f0fda08aSwikidesign * Outputs the comment form 792f0fda08aSwikidesign */ 793f0fda08aSwikidesign function _form($raw = '', $act = 'add', $cid = NULL) { 7943011fb8bSMichael Klier global $lang; 7953011fb8bSMichael Klier global $conf; 7963011fb8bSMichael Klier global $ID; 7973011fb8bSMichael Klier global $INFO; 798f0fda08aSwikidesign 799f0fda08aSwikidesign // not for unregistered users when guest comments aren't allowed 8007c8e18ffSGina Haeussge if (!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) { 8017c8e18ffSGina Haeussge ?> 8027c8e18ffSGina Haeussge <div class="comment_form"> 8037c8e18ffSGina Haeussge <?php echo $this->getLang('noguests'); ?> 8047c8e18ffSGina Haeussge </div> 8057c8e18ffSGina Haeussge <?php 8067c8e18ffSGina Haeussge return false; 8077c8e18ffSGina Haeussge } 808f0fda08aSwikidesign 8091ba72c23Swikidesign // fill $raw with $_REQUEST['text'] if it's empty (for failed CAPTCHA check) 8101ba72c23Swikidesign if (!$raw && ($_REQUEST['comment'] == 'show')) $raw = $_REQUEST['text']; 811f0fda08aSwikidesign ?> 8125ef1705fSiLoveiDo 813f0fda08aSwikidesign <div class="comment_form"> 814bea69264SMichael Klier <form id="discussion__comment_form" method="post" action="<?php echo script() ?>" accept-charset="<?php echo $lang['encoding'] ?>"> 815f0fda08aSwikidesign <div class="no"> 816f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 81761437513Swikidesign <input type="hidden" name="do" value="show" /> 818f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 819530693fbSMichael Klier <?php 820f0fda08aSwikidesign // for adding a comment 821f0fda08aSwikidesign if ($act == 'add') { 822f0fda08aSwikidesign ?> 823f0fda08aSwikidesign <input type="hidden" name="reply" value="<?php echo $cid ?>" /> 824f0fda08aSwikidesign <?php 82520c152acSMichael Klier // for guest/adminimport: show name, e-mail and subscribe to comments fields 82620c152acSMichael Klier if(!$_SERVER['REMOTE_USER'] or ($this->getConf('adminimport') && auth_ismanager())) { 827f0fda08aSwikidesign ?> 828f0fda08aSwikidesign <input type="hidden" name="user" value="<?php echo clientIP() ?>" /> 829f0fda08aSwikidesign <div class="comment_name"> 830f0fda08aSwikidesign <label class="block" for="discussion__comment_name"> 831f0fda08aSwikidesign <span><?php echo $lang['fullname'] ?>:</span> 83237e3c825SMichael 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'])?>" /> 833f0fda08aSwikidesign </label> 834f0fda08aSwikidesign </div> 835f0fda08aSwikidesign <div class="comment_mail"> 836f0fda08aSwikidesign <label class="block" for="discussion__comment_mail"> 837f0fda08aSwikidesign <span><?php echo $lang['email'] ?>:</span> 83837e3c825SMichael 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'])?>" /> 839f0fda08aSwikidesign </label> 840f0fda08aSwikidesign </div> 841f0fda08aSwikidesign <?php 842f0fda08aSwikidesign } 843f0fda08aSwikidesign 844f0fda08aSwikidesign // allow entering an URL 845f0fda08aSwikidesign if ($this->getConf('urlfield')) { 846f0fda08aSwikidesign ?> 847f0fda08aSwikidesign <div class="comment_url"> 848f0fda08aSwikidesign <label class="block" for="discussion__comment_url"> 849f0fda08aSwikidesign <span><?php echo $this->getLang('url') ?>:</span> 850e7c760b3Swikidesign <input type="text" class="edit" name="url" id="discussion__comment_url" size="50" tabindex="3" value="<?php echo hsc($_REQUEST['url'])?>" /> 851f0fda08aSwikidesign </label> 852f0fda08aSwikidesign </div> 853f0fda08aSwikidesign <?php 854f0fda08aSwikidesign } 855f0fda08aSwikidesign 856f0fda08aSwikidesign // allow entering an address 857f0fda08aSwikidesign if ($this->getConf('addressfield')) { 858f0fda08aSwikidesign ?> 859f0fda08aSwikidesign <div class="comment_address"> 860f0fda08aSwikidesign <label class="block" for="discussion__comment_address"> 861f0fda08aSwikidesign <span><?php echo $this->getLang('address') ?>:</span> 862e7c760b3Swikidesign <input type="text" class="edit" name="address" id="discussion__comment_address" size="50" tabindex="4" value="<?php echo hsc($_REQUEST['address'])?>" /> 863f0fda08aSwikidesign </label> 864f0fda08aSwikidesign </div> 865f0fda08aSwikidesign <?php 866f0fda08aSwikidesign } 867f0fda08aSwikidesign 868f0fda08aSwikidesign // allow setting the comment date 869a1ace3c9Swikidesign if ($this->getConf('adminimport') && (auth_ismanager())) { 870f0fda08aSwikidesign ?> 871f0fda08aSwikidesign <div class="comment_date"> 872f0fda08aSwikidesign <label class="block" for="discussion__comment_date"> 873f0fda08aSwikidesign <span><?php echo $this->getLang('date') ?>:</span> 874f0fda08aSwikidesign <input type="text" class="edit" name="date" id="discussion__comment_date" size="50" /> 875f0fda08aSwikidesign </label> 876f0fda08aSwikidesign </div> 877f0fda08aSwikidesign <?php 878f0fda08aSwikidesign } 879f0fda08aSwikidesign 880f0fda08aSwikidesign // for saving a comment 881f0fda08aSwikidesign } else { 882f0fda08aSwikidesign ?> 883f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 884f0fda08aSwikidesign <?php 885f0fda08aSwikidesign } 886f0fda08aSwikidesign ?> 887f0fda08aSwikidesign <div class="comment_text"> 8881de52da1SMichael Klier <div id="discussion__comment_toolbar"> 8891de52da1SMichael Klier <?php echo $this->getLang('entercomment')?> 8901de52da1SMichael Klier <?php if($this->getLang('wikisyntaxok')) echo ', ' . $this->getLang('wikisyntax') . ':';?> 8911de52da1SMichael Klier </div> 89237e3c825SMichael 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 89337e3c825SMichael Klier if($raw) { 89437e3c825SMichael Klier echo formText($raw); 89537e3c825SMichael Klier } else { 89637e3c825SMichael Klier echo $_REQUEST['text']; 89737e3c825SMichael Klier } 89837e3c825SMichael Klier ?></textarea> 899f0fda08aSwikidesign </div> 900e7c760b3Swikidesign <?php //bad and dirty event insert hook 901e7c760b3Swikidesign $evdata = array('writable' => true); 902e7c760b3Swikidesign trigger_event('HTML_EDITFORM_INJECTION', $evdata); 903e7c760b3Swikidesign ?> 904bea69264SMichael 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" /> 9050c3df1b0SMichael 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]" /> 9063011fb8bSMichael Klier 907189071d2SMichael Klier <?php if((!$_SERVER['REMOTE_USER'] || $_SERVER['REMOTE_USER'] && !$conf['subscribers']) && $this->getConf('subscribe')) { ?> 9083011fb8bSMichael Klier <div class="comment_subscribe"> 90911b1339eSGina Haeussge <input type="checkbox" id="discussion__comment_subscribe" name="subscribe" tabindex="6" /> 9103011fb8bSMichael Klier <label class="block" for="discussion__comment_subscribe"> 9113011fb8bSMichael Klier <span><?php echo $this->getLang('subscribe') ?></span> 9123011fb8bSMichael Klier </label> 9133011fb8bSMichael Klier </div> 9143011fb8bSMichael Klier <?php } ?> 9153011fb8bSMichael Klier 9163011fb8bSMichael Klier <div class="clearer"></div> 917ed4846efSMichael Klier <div id="discussion__comment_preview"> </div> 918f0fda08aSwikidesign </div> 919f0fda08aSwikidesign </form> 920f0fda08aSwikidesign </div> 921f0fda08aSwikidesign <?php 922f0fda08aSwikidesign if ($this->getConf('usecocomment')) echo $this->_coComment(); 923f0fda08aSwikidesign } 924f0fda08aSwikidesign 925f0fda08aSwikidesign /** 926f0fda08aSwikidesign * Adds a javascript to interact with coComments 927f0fda08aSwikidesign */ 928f0fda08aSwikidesign function _coComment() { 9292b18adb9SMichael Klier global $ID; 9302b18adb9SMichael Klier global $conf; 9312b18adb9SMichael Klier global $INFO; 932f0fda08aSwikidesign 933f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 934f0fda08aSwikidesign 935f0fda08aSwikidesign ?> 936f0fda08aSwikidesign <script type="text/javascript"><!--//--><![CDATA[//><!-- 937f0fda08aSwikidesign var blogTool = "DokuWiki"; 938f0fda08aSwikidesign var blogURL = "<?php echo DOKU_URL ?>"; 939f0fda08aSwikidesign var blogTitle = "<?php echo $conf['title'] ?>"; 940f0fda08aSwikidesign var postURL = "<?php echo wl($ID, '', true) ?>"; 941f0fda08aSwikidesign var postTitle = "<?php echo tpl_pagetitle($ID, true) ?>"; 942f0fda08aSwikidesign <?php 943f0fda08aSwikidesign if ($user) { 944f0fda08aSwikidesign ?> 945f0fda08aSwikidesign var commentAuthor = "<?php echo $INFO['userinfo']['name'] ?>"; 946f0fda08aSwikidesign <?php 947f0fda08aSwikidesign } else { 948f0fda08aSwikidesign ?> 949f0fda08aSwikidesign var commentAuthorFieldName = "name"; 950f0fda08aSwikidesign <?php 951f0fda08aSwikidesign } 952f0fda08aSwikidesign ?> 953f0fda08aSwikidesign var commentAuthorLoggedIn = <?php echo ($user ? 'true' : 'false') ?>; 954f0fda08aSwikidesign var commentFormID = "discussion__comment_form"; 955f0fda08aSwikidesign var commentTextFieldName = "text"; 956f0fda08aSwikidesign var commentButtonName = "submit"; 957f0fda08aSwikidesign var cocomment_force = false; 958f0fda08aSwikidesign //--><!]]></script> 959f0fda08aSwikidesign <script type="text/javascript" src="http://www.cocomment.com/js/cocomment.js"> 960f0fda08aSwikidesign </script> 961f0fda08aSwikidesign <?php 962f0fda08aSwikidesign } 963f0fda08aSwikidesign 964f0fda08aSwikidesign /** 965f0fda08aSwikidesign * General button function 966f0fda08aSwikidesign */ 9671e46d176Swikidesign function _button($cid, $label, $act, $jump = false) { 968f0fda08aSwikidesign global $ID; 9695ef1705fSiLoveiDo 9701e46d176Swikidesign $anchor = ($jump ? '#discussion__comment_form' : '' ); 971f0fda08aSwikidesign 972f0fda08aSwikidesign ?> 9736d1f4f20SGina Haeussge <form class="button discussion__<?php echo $act?>" method="get" action="<?php echo script().$anchor ?>"> 974f0fda08aSwikidesign <div class="no"> 975f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 97661437513Swikidesign <input type="hidden" name="do" value="show" /> 977f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 978f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 979f0fda08aSwikidesign <input type="submit" value="<?php echo $label ?>" class="button" title="<?php echo $label ?>" /> 980f0fda08aSwikidesign </div> 981f0fda08aSwikidesign </form> 982f0fda08aSwikidesign <?php 983f0fda08aSwikidesign return true; 984f0fda08aSwikidesign } 985f0fda08aSwikidesign 986f0fda08aSwikidesign /** 987f0fda08aSwikidesign * Adds an entry to the comments changelog 988f0fda08aSwikidesign * 989f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 990f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 991f0fda08aSwikidesign */ 992f0fda08aSwikidesign function _addLogEntry($date, $id, $type = 'cc', $summary = '', $extra = '') { 993f0fda08aSwikidesign global $conf; 994f0fda08aSwikidesign 995f0fda08aSwikidesign $changelog = $conf['metadir'].'/_comments.changes'; 996f0fda08aSwikidesign 997f0fda08aSwikidesign if(!$date) $date = time(); //use current time if none supplied 998f0fda08aSwikidesign $remote = $_SERVER['REMOTE_ADDR']; 999f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 1000f0fda08aSwikidesign 1001f0fda08aSwikidesign $strip = array("\t", "\n"); 1002f0fda08aSwikidesign $logline = array( 1003f0fda08aSwikidesign 'date' => $date, 1004f0fda08aSwikidesign 'ip' => $remote, 1005f0fda08aSwikidesign 'type' => str_replace($strip, '', $type), 1006f0fda08aSwikidesign 'id' => $id, 1007f0fda08aSwikidesign 'user' => $user, 1008f0fda08aSwikidesign 'sum' => str_replace($strip, '', $summary), 1009f0fda08aSwikidesign 'extra' => str_replace($strip, '', $extra) 1010f0fda08aSwikidesign ); 1011f0fda08aSwikidesign 1012f0fda08aSwikidesign // add changelog line 1013f0fda08aSwikidesign $logline = implode("\t", $logline)."\n"; 1014f0fda08aSwikidesign io_saveFile($changelog, $logline, true); //global changelog cache 1015f0fda08aSwikidesign $this->_trimRecentCommentsLog($changelog); 101677a22ba2Swikidesign 101777a22ba2Swikidesign // tell the indexer to re-index the page 101877a22ba2Swikidesign @unlink(metaFN($id, '.indexed')); 1019f0fda08aSwikidesign } 1020f0fda08aSwikidesign 1021f0fda08aSwikidesign /** 1022f0fda08aSwikidesign * Trims the recent comments cache to the last $conf['changes_days'] recent 1023f0fda08aSwikidesign * changes or $conf['recent'] items, which ever is larger. 1024f0fda08aSwikidesign * The trimming is only done once a day. 1025f0fda08aSwikidesign * 1026f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 1027f0fda08aSwikidesign */ 1028f0fda08aSwikidesign function _trimRecentCommentsLog($changelog) { 1029f0fda08aSwikidesign global $conf; 1030f0fda08aSwikidesign 1031f0fda08aSwikidesign if (@file_exists($changelog) && 1032f0fda08aSwikidesign (filectime($changelog) + 86400) < time() && 1033f0fda08aSwikidesign !@file_exists($changelog.'_tmp')) { 1034f0fda08aSwikidesign 1035f0fda08aSwikidesign io_lock($changelog); 1036f0fda08aSwikidesign $lines = file($changelog); 1037f0fda08aSwikidesign if (count($lines)<$conf['recent']) { 1038f0fda08aSwikidesign // nothing to trim 1039f0fda08aSwikidesign io_unlock($changelog); 1040f0fda08aSwikidesign return true; 1041f0fda08aSwikidesign } 1042f0fda08aSwikidesign 1043f0fda08aSwikidesign io_saveFile($changelog.'_tmp', ''); // presave tmp as 2nd lock 1044f0fda08aSwikidesign $trim_time = time() - $conf['recent_days']*86400; 1045f0fda08aSwikidesign $out_lines = array(); 1046f0fda08aSwikidesign 1047e49085a2SMichael Klier $num = count($lines); 1048e49085a2SMichael Klier for ($i=0; $i<$num; $i++) { 1049f0fda08aSwikidesign $log = parseChangelogLine($lines[$i]); 1050f0fda08aSwikidesign if ($log === false) continue; // discard junk 1051f0fda08aSwikidesign if ($log['date'] < $trim_time) { 1052f0fda08aSwikidesign $old_lines[$log['date'].".$i"] = $lines[$i]; // keep old lines for now (append .$i to prevent key collisions) 1053f0fda08aSwikidesign } else { 1054f0fda08aSwikidesign $out_lines[$log['date'].".$i"] = $lines[$i]; // definitely keep these lines 1055f0fda08aSwikidesign } 1056f0fda08aSwikidesign } 1057f0fda08aSwikidesign 1058f0fda08aSwikidesign // sort the final result, it shouldn't be necessary, 1059f0fda08aSwikidesign // however the extra robustness in making the changelog cache self-correcting is worth it 1060f0fda08aSwikidesign ksort($out_lines); 1061f0fda08aSwikidesign $extra = $conf['recent'] - count($out_lines); // do we need extra lines do bring us up to minimum 1062f0fda08aSwikidesign if ($extra > 0) { 1063f0fda08aSwikidesign ksort($old_lines); 1064f0fda08aSwikidesign $out_lines = array_merge(array_slice($old_lines,-$extra),$out_lines); 1065f0fda08aSwikidesign } 1066f0fda08aSwikidesign 1067f0fda08aSwikidesign // save trimmed changelog 1068f0fda08aSwikidesign io_saveFile($changelog.'_tmp', implode('', $out_lines)); 1069f0fda08aSwikidesign @unlink($changelog); 1070f0fda08aSwikidesign if (!rename($changelog.'_tmp', $changelog)) { 1071f0fda08aSwikidesign // rename failed so try another way... 1072f0fda08aSwikidesign io_unlock($changelog); 1073f0fda08aSwikidesign io_saveFile($changelog, implode('', $out_lines)); 1074f0fda08aSwikidesign @unlink($changelog.'_tmp'); 1075f0fda08aSwikidesign } else { 1076f0fda08aSwikidesign io_unlock($changelog); 1077f0fda08aSwikidesign } 1078f0fda08aSwikidesign return true; 1079f0fda08aSwikidesign } 1080f0fda08aSwikidesign } 1081f0fda08aSwikidesign 1082f0fda08aSwikidesign /** 1083f0fda08aSwikidesign * Sends a notify mail on new comment 1084f0fda08aSwikidesign * 1085f0fda08aSwikidesign * @param array $comment data array of the new comment 1086f0fda08aSwikidesign * 1087f0fda08aSwikidesign * @author Andreas Gohr <andi@splitbrain.org> 1088f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 1089f0fda08aSwikidesign */ 10909881d835SMichael Klier function _notify($comment, &$subscribers) { 1091f0fda08aSwikidesign global $conf; 1092f0fda08aSwikidesign global $ID; 1093bd15cd45SAndy Webber global $INFO; 1094f0fda08aSwikidesign 10959881d835SMichael Klier $notify_text = io_readfile($this->localfn('subscribermail')); 10969881d835SMichael Klier $confirm_text = io_readfile($this->localfn('confirmsubscribe')); 10979881d835SMichael Klier $subject_notify = '['.$conf['title'].'] '.$this->getLang('mail_newcomment'); 10989881d835SMichael Klier $subject_subscribe = '['.$conf['title'].'] '.$this->getLang('subscribe'); 1099bd15cd45SAndy Webber $from = $conf['mailfrom']; 1100bd15cd45SAndy Webber $from = str_replace('@USER@',$_SERVER['REMOTE_USER'],$from); 1101bd15cd45SAndy Webber $from = str_replace('@NAME@',$INFO['userinfo']['name'],$from); 1102bd15cd45SAndy Webber $from = str_replace('@MAIL@',$INFO['userinfo']['mail'],$from); 1103f0fda08aSwikidesign 11046046f25cSwikidesign $search = array( 11056046f25cSwikidesign '@PAGE@', 11066046f25cSwikidesign '@TITLE@', 11076046f25cSwikidesign '@DATE@', 11086046f25cSwikidesign '@NAME@', 11096046f25cSwikidesign '@TEXT@', 11108b42cefbSMichael Klier '@COMMENTURL@', 11116046f25cSwikidesign '@UNSUBSCRIBE@', 11126046f25cSwikidesign '@DOKUWIKIURL@', 11136046f25cSwikidesign ); 11143011fb8bSMichael Klier 11153011fb8bSMichael Klier // notify page subscribers 11168c6e87cfSMichael Klier if ($conf['subscribers'] || $conf['notify']) { 111705eeb3afSJulian Knauer $list = explode(',', subscription_addresslist($ID)); 11188c6e87cfSMichael Klier $to = (!empty($conf['notify'])) ? $conf['notify'] : array_pop($list); 1119189071d2SMichael Klier $bcc = implode(',', $list); 11203011fb8bSMichael Klier 11216046f25cSwikidesign $replace = array( 11226046f25cSwikidesign $ID, 11236046f25cSwikidesign $conf['title'], 1124f014bc86SMichael Klier strftime($conf['dformat'], $comment['date']['created']), 11256046f25cSwikidesign $comment['user']['name'], 11266046f25cSwikidesign $comment['raw'], 11271810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 11286046f25cSwikidesign wl($ID, 'do=unsubscribe', true, '&'), 11296046f25cSwikidesign DOKU_URL, 11306046f25cSwikidesign ); 1131f0fda08aSwikidesign 11329881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 1133bd15cd45SAndy Webber mail_send($to, $subject_notify, $body, $from, '', $bcc); 11343011fb8bSMichael Klier } 1135f0fda08aSwikidesign 11363011fb8bSMichael Klier // notify comment subscribers 11373011fb8bSMichael Klier if (!empty($subscribers)) { 11383011fb8bSMichael Klier 11399881d835SMichael Klier foreach($subscribers as $mail => $data) { 11403011fb8bSMichael Klier $to = $mail; 11413011fb8bSMichael Klier 11429881d835SMichael Klier if($data['active']) { 11433011fb8bSMichael Klier $replace = array( 11443011fb8bSMichael Klier $ID, 11453011fb8bSMichael Klier $conf['title'], 11463011fb8bSMichael Klier strftime($conf['dformat'], $comment['date']['created']), 11473011fb8bSMichael Klier $comment['user']['name'], 11483011fb8bSMichael Klier $comment['raw'], 11491810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 1150b2f2e866SMichael Klier wl($ID, 'do=discussion_unsubscribe&hash=' . $data['hash'], true, '&'), 11513011fb8bSMichael Klier DOKU_URL, 11523011fb8bSMichael Klier ); 11533011fb8bSMichael Klier 11549881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 1155bd15cd45SAndy Webber mail_send($to, $subject_notify, $body, $from); 11569881d835SMichael Klier } elseif(!$data['active'] && !$data['confirmsent']) { 11579881d835SMichael Klier $search = array( 11589881d835SMichael Klier '@PAGE@', 11599881d835SMichael Klier '@TITLE@', 11609881d835SMichael Klier '@SUBSCRIBE@', 11619881d835SMichael Klier '@DOKUWIKIURL@', 11629881d835SMichael Klier ); 11639881d835SMichael Klier $replace = array( 11649881d835SMichael Klier $ID, 11659881d835SMichael Klier $conf['title'], 1166b2f2e866SMichael Klier wl($ID, 'do=discussion_confirmsubscribe&hash=' . $data['hash'], true, '&'), 11679881d835SMichael Klier DOKU_URL, 11689881d835SMichael Klier ); 11699881d835SMichael Klier 11709881d835SMichael Klier $body = str_replace($search, $replace, $confirm_text); 1171bd15cd45SAndy Webber mail_send($to, $subject_subscribe, $body, $from); 11729881d835SMichael Klier $subscribers[$mail]['confirmsent'] = true; 11739881d835SMichael Klier } 11743011fb8bSMichael Klier } 11753011fb8bSMichael Klier } 1176f0fda08aSwikidesign } 1177f0fda08aSwikidesign 1178f0fda08aSwikidesign /** 1179f0fda08aSwikidesign * Counts the number of visible comments 1180f0fda08aSwikidesign */ 1181f0fda08aSwikidesign function _count($data) { 1182f0fda08aSwikidesign $number = 0; 1183f0fda08aSwikidesign foreach ($data['comments'] as $cid => $comment) { 1184f0fda08aSwikidesign if ($comment['parent']) continue; 1185f0fda08aSwikidesign if (!$comment['show']) continue; 1186f0fda08aSwikidesign $number++; 1187f0fda08aSwikidesign $rids = $comment['replies']; 1188f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1189f0fda08aSwikidesign } 1190f0fda08aSwikidesign return $number; 1191f0fda08aSwikidesign } 1192f0fda08aSwikidesign 1193f0fda08aSwikidesign function _countReplies(&$data, $rids) { 1194f0fda08aSwikidesign $number = 0; 1195f0fda08aSwikidesign foreach ($rids as $rid) { 11962ee3dca3Swikidesign if (!isset($data['comments'][$rid])) continue; // reply was removed 1197f0fda08aSwikidesign if (!$data['comments'][$rid]['show']) continue; 1198f0fda08aSwikidesign $number++; 1199f0fda08aSwikidesign $rids = $data['comments'][$rid]['replies']; 1200f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1201f0fda08aSwikidesign } 1202f0fda08aSwikidesign return $number; 1203f0fda08aSwikidesign } 1204f0fda08aSwikidesign 1205f0fda08aSwikidesign /** 1206f0fda08aSwikidesign * Renders the comment text 1207f0fda08aSwikidesign */ 1208f0fda08aSwikidesign function _render($raw) { 1209f0fda08aSwikidesign if ($this->getConf('wikisyntaxok')) { 1210f0fda08aSwikidesign $xhtml = $this->render($raw); 1211f0fda08aSwikidesign } else { // wiki syntax not allowed -> just encode special chars 121287bb4e97SMichael Klier $xhtml = hsc(trim($raw)); 121387bb4e97SMichael Klier $xhtml = str_replace("\n", '<br />', $xhtml); 1214f0fda08aSwikidesign } 1215f0fda08aSwikidesign return $xhtml; 1216f0fda08aSwikidesign } 1217f0fda08aSwikidesign 1218f0fda08aSwikidesign /** 1219479dd10fSwikidesign * Finds out whether there is a discussion section for the current page 1220479dd10fSwikidesign */ 1221e6a89be2Swikidesign function _hasDiscussion(&$title) { 1222b2ac3b3bSwikidesign global $ID; 12234a0a1bd2Swikidesign 1224479dd10fSwikidesign $cfile = metaFN($ID, '.comments'); 1225479dd10fSwikidesign 1226479dd10fSwikidesign if (!@file_exists($cfile)) { 12272b18adb9SMichael Klier if ($this->getConf('automatic')) { 12282b18adb9SMichael Klier return true; 12292b18adb9SMichael Klier } else { 12302b18adb9SMichael Klier return false; 12312b18adb9SMichael Klier } 1232479dd10fSwikidesign } 1233479dd10fSwikidesign 1234479dd10fSwikidesign $comments = unserialize(io_readFile($cfile, false)); 1235479dd10fSwikidesign 123607c376bbSwikidesign if ($comments['title']) $title = hsc($comments['title']); 1237479dd10fSwikidesign $num = $comments['number']; 1238479dd10fSwikidesign if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return false; 1239479dd10fSwikidesign else return true; 1240479dd10fSwikidesign } 1241479dd10fSwikidesign 1242479dd10fSwikidesign /** 1243e7c760b3Swikidesign * Creates a new thread page 1244e7c760b3Swikidesign */ 12456046f25cSwikidesign function _newThread() { 124661437513Swikidesign global $ID, $INFO; 1247f0fda08aSwikidesign 12481ea794e5Swikidesign $ns = cleanID($_REQUEST['ns']); 1249f0fda08aSwikidesign $title = str_replace(':', '', $_REQUEST['title']); 12502e80cd5fSwikidesign $back = $ID; 12512e80cd5fSwikidesign $ID = ($ns ? $ns.':' : '').cleanID($title); 12522e80cd5fSwikidesign $INFO = pageinfo(); 1253f0fda08aSwikidesign 1254f0fda08aSwikidesign // check if we are allowed to create this file 12552e80cd5fSwikidesign if ($INFO['perm'] >= AUTH_CREATE) { 1256f0fda08aSwikidesign 1257f0fda08aSwikidesign //check if locked by anyone - if not lock for my self 12582e80cd5fSwikidesign if ($INFO['locked']) return 'locked'; 12592e80cd5fSwikidesign else lock($ID); 1260f0fda08aSwikidesign 1261f0fda08aSwikidesign // prepare the new thread file with default stuff 12622e80cd5fSwikidesign if (!@file_exists($INFO['filepath'])) { 1263f0fda08aSwikidesign global $TEXT; 1264f0fda08aSwikidesign 12652e80cd5fSwikidesign $TEXT = pageTemplate(array(($ns ? $ns.':' : '').$title)); 12661433886fSwikidesign if (!$TEXT) { 126761437513Swikidesign $data = array('id' => $ID, 'ns' => $ns, 'title' => $title, 'back' => $back); 126861437513Swikidesign $TEXT = $this->_pageTemplate($data); 12692e80cd5fSwikidesign } 12702e80cd5fSwikidesign return 'preview'; 1271f0fda08aSwikidesign } else { 12722e80cd5fSwikidesign return 'edit'; 1273f0fda08aSwikidesign } 1274f0fda08aSwikidesign } else { 12752e80cd5fSwikidesign return 'show'; 1276f0fda08aSwikidesign } 1277f0fda08aSwikidesign } 1278f0fda08aSwikidesign 1279e7c760b3Swikidesign /** 128061437513Swikidesign * Adapted version of pageTemplate() function 128161437513Swikidesign */ 128261437513Swikidesign function _pageTemplate($data) { 128361437513Swikidesign global $conf, $INFO; 128461437513Swikidesign 128561437513Swikidesign $id = $data['id']; 128661437513Swikidesign $user = $_SERVER['REMOTE_USER']; 128761437513Swikidesign $tpl = io_readFile(DOKU_PLUGIN.'discussion/_template.txt'); 128861437513Swikidesign 128961437513Swikidesign // standard replacements 129061437513Swikidesign $replace = array( 129161437513Swikidesign '@NS@' => $data['ns'], 129261437513Swikidesign '@PAGE@' => strtr(noNS($id),'_',' '), 129361437513Swikidesign '@USER@' => $user, 129461437513Swikidesign '@NAME@' => $INFO['userinfo']['name'], 129561437513Swikidesign '@MAIL@' => $INFO['userinfo']['mail'], 1296f014bc86SMichael Klier '@DATE@' => strftime($conf['dformat']), 129761437513Swikidesign ); 129861437513Swikidesign 129961437513Swikidesign // additional replacements 130061437513Swikidesign $replace['@BACK@'] = $data['back']; 130161437513Swikidesign $replace['@TITLE@'] = $data['title']; 130261437513Swikidesign 130361437513Swikidesign // avatar if useavatar and avatar plugin available 130461437513Swikidesign if ($this->getConf('useavatar') 130561437513Swikidesign && (@file_exists(DOKU_PLUGIN.'avatar/syntax.php')) 130661437513Swikidesign && (!plugin_isdisabled('avatar'))) { 130761437513Swikidesign $replace['@AVATAR@'] = '{{avatar>'.$user.' }} '; 130861437513Swikidesign } else { 130961437513Swikidesign $replace['@AVATAR@'] = ''; 131061437513Swikidesign } 131161437513Swikidesign 131261437513Swikidesign // tag if tag plugin is available 131361437513Swikidesign if ((@file_exists(DOKU_PLUGIN.'tag/syntax/tag.php')) 131461437513Swikidesign && (!plugin_isdisabled('tag'))) { 131561437513Swikidesign $replace['@TAG@'] = "\n\n{{tag>}}"; 131661437513Swikidesign } else { 131761437513Swikidesign $replace['@TAG@'] = ''; 131861437513Swikidesign } 131961437513Swikidesign 132061437513Swikidesign // do the replace 132161437513Swikidesign $tpl = str_replace(array_keys($replace), array_values($replace), $tpl); 132261437513Swikidesign return $tpl; 132361437513Swikidesign } 132461437513Swikidesign 132561437513Swikidesign /** 1326e7c760b3Swikidesign * Checks if the CAPTCHA string submitted is valid 1327e7c760b3Swikidesign * 1328e7c760b3Swikidesign * @author Andreas Gohr <gohr@cosmocode.de> 1329e7c760b3Swikidesign * @adaption Esther Brunner <wikidesign@gmail.com> 1330e7c760b3Swikidesign */ 13316046f25cSwikidesign function _captchaCheck() { 1332be1d1f2fSGina Haeussge if (plugin_isdisabled('captcha') || (!$captcha = plugin_load('helper', 'captcha'))) 1333be1d1f2fSGina Haeussge return; // CAPTCHA is disabled or not available 1334e7c760b3Swikidesign 1335d1c29589Swikidesign // do nothing if logged in user and no CAPTCHA required 1336d1c29589Swikidesign if (!$captcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return; 1337d1c29589Swikidesign 1338e7c760b3Swikidesign // compare provided string with decrypted captcha 1339e7c760b3Swikidesign $rand = PMA_blowfish_decrypt($_REQUEST['plugin__captcha_secret'], auth_cookiesalt()); 1340e7c760b3Swikidesign $code = $captcha->_generateCAPTCHA($captcha->_fixedIdent(), $rand); 1341e7c760b3Swikidesign 1342e7c760b3Swikidesign if (!$_REQUEST['plugin__captcha_secret'] || 1343e7c760b3Swikidesign !$_REQUEST['plugin__captcha'] || 1344e7c760b3Swikidesign strtoupper($_REQUEST['plugin__captcha']) != $code) { 1345e7c760b3Swikidesign 1346e7c760b3Swikidesign // CAPTCHA test failed! Continue to edit instead of saving 1347e7c760b3Swikidesign msg($captcha->getLang('testfailed'), -1); 1348e7c760b3Swikidesign if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit'; 1349e7c760b3Swikidesign elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show'; 1350e7c760b3Swikidesign } 1351e7c760b3Swikidesign // if we arrive here it was a valid save 1352e7c760b3Swikidesign } 1353e7c760b3Swikidesign 1354a1ca9e44Swikidesign /** 1355bd6dc08eSAdrian Schlegel * checks if the submitted reCAPTCHA string is valid 1356bd6dc08eSAdrian Schlegel * 1357bd6dc08eSAdrian Schlegel * @author Adrian Schlegel <adrian@liip.ch> 1358bd6dc08eSAdrian Schlegel */ 1359bd6dc08eSAdrian Schlegel function _recaptchaCheck() { 1360bd6dc08eSAdrian Schlegel if (plugin_isdisabled('recaptcha') || (!$recaptcha = plugin_load('helper', 'recaptcha'))) 1361bd6dc08eSAdrian Schlegel return; // reCAPTCHA is disabled or not available 1362bd6dc08eSAdrian Schlegel 1363bd6dc08eSAdrian Schlegel // do nothing if logged in user and no reCAPTCHA required 1364bd6dc08eSAdrian Schlegel if (!$recaptcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return; 1365bd6dc08eSAdrian Schlegel 1366bd6dc08eSAdrian Schlegel $resp = $recaptcha->check(); 1367bd6dc08eSAdrian Schlegel if (!$resp->is_valid) { 1368bd6dc08eSAdrian Schlegel msg($recaptcha->getLang('testfailed'),-1); 1369bd6dc08eSAdrian Schlegel if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit'; 1370bd6dc08eSAdrian Schlegel elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show'; 1371bd6dc08eSAdrian Schlegel } 1372bd6dc08eSAdrian Schlegel } 1373bd6dc08eSAdrian Schlegel 1374bd6dc08eSAdrian Schlegel /** 1375a1ca9e44Swikidesign * Adds the comments to the index 1376a1ca9e44Swikidesign */ 137761437513Swikidesign function idx_add_discussion(&$event, $param) { 1378a1ca9e44Swikidesign 1379a1ca9e44Swikidesign // get .comments meta file name 1380a1ca9e44Swikidesign $file = metaFN($event->data[0], '.comments'); 1381a1ca9e44Swikidesign 1382a1ca9e44Swikidesign if (@file_exists($file)) $data = unserialize(io_readFile($file, false)); 1383a1ca9e44Swikidesign if ((!$data['status']) || ($data['number'] == 0)) return; // comments are turned off 1384a1ca9e44Swikidesign 1385a1ca9e44Swikidesign // now add the comments 1386a1ca9e44Swikidesign if (isset($data['comments'])) { 1387a1ca9e44Swikidesign foreach ($data['comments'] as $key => $value) { 138877a22ba2Swikidesign $event->data[1] .= $this->_addCommentWords($key, $data); 1389a1ca9e44Swikidesign } 1390a1ca9e44Swikidesign } 1391a1ca9e44Swikidesign } 1392a1ca9e44Swikidesign 1393a1ca9e44Swikidesign /** 1394a1ca9e44Swikidesign * Adds the words of a given comment to the index 1395a1ca9e44Swikidesign */ 1396efbe59d0Swikidesign function _addCommentWords($cid, &$data, $parent = '') { 1397a1ca9e44Swikidesign 1398efbe59d0Swikidesign if (!isset($data['comments'][$cid])) return ''; // comment was removed 1399a1ca9e44Swikidesign $comment = $data['comments'][$cid]; 1400a1ca9e44Swikidesign 1401efbe59d0Swikidesign if (!is_array($comment)) return ''; // corrupt datatype 1402efbe59d0Swikidesign if ($comment['parent'] != $parent) return ''; // reply to an other comment 1403efbe59d0Swikidesign if (!$comment['show']) return ''; // hidden comment 1404a1ca9e44Swikidesign 1405efbe59d0Swikidesign $text = $comment['raw']; // we only add the raw comment text 1406efbe59d0Swikidesign if (is_array($comment['replies'])) { // and the replies 1407efbe59d0Swikidesign foreach ($comment['replies'] as $rid) { 1408efbe59d0Swikidesign $text .= $this->_addCommentWords($rid, $data, $cid); 1409a1ca9e44Swikidesign } 1410a1ca9e44Swikidesign } 1411efbe59d0Swikidesign return ' '.$text; 1412efbe59d0Swikidesign } 1413a10b5c98SMichael Klier 1414a10b5c98SMichael Klier /** 1415a10b5c98SMichael Klier * Only allow http(s) URLs and append http:// to URLs if needed 1416a10b5c98SMichael Klier */ 1417a10b5c98SMichael Klier function _checkURL($url) { 1418a10b5c98SMichael Klier if(preg_match("#^http://|^https://#", $url)) { 1419a10b5c98SMichael Klier return hsc($url); 1420a10b5c98SMichael Klier } elseif(substr($url, 0, 4) == 'www.') { 1421a10b5c98SMichael Klier return hsc('http://' . $url); 1422a10b5c98SMichael Klier } else { 1423a10b5c98SMichael Klier return ''; 1424a10b5c98SMichael Klier } 1425a10b5c98SMichael Klier } 1426f0fda08aSwikidesign} 142731aab30eSGina Haeussge 142831aab30eSGina Haeussgefunction _sortCallback($a, $b) { 142931aab30eSGina Haeussge if (is_array($a['date'])) { // new format 143031aab30eSGina Haeussge $createdA = $a['date']['created']; 143131aab30eSGina Haeussge } else { // old format 143231aab30eSGina Haeussge $createdA = $a['date']; 143331aab30eSGina Haeussge } 143431aab30eSGina Haeussge 143531aab30eSGina Haeussge if (is_array($b['date'])) { // new format 143631aab30eSGina Haeussge $createdB = $b['date']['created']; 143731aab30eSGina Haeussge } else { // old format 143831aab30eSGina Haeussge $createdB = $b['date']; 143931aab30eSGina Haeussge } 144031aab30eSGina Haeussge 144131aab30eSGina Haeussge if ($createdA == $createdB) 144231aab30eSGina Haeussge return 0; 144331aab30eSGina Haeussge else 144431aab30eSGina Haeussge return ($createdA < $createdB) ? -1 : 1; 144531aab30eSGina Haeussge} 144631aab30eSGina Haeussge 1447530693fbSMichael Klier// vim:ts=4:sw=4:et:enc=utf-8: 1448