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 ); 76b8fdc796SMichael Klier } 77b8fdc796SMichael Klier 78b8fdc796SMichael Klier /** 79b8fdc796SMichael Klier * Preview Comments 80b8fdc796SMichael Klier * 81b8fdc796SMichael Klier * @author Michael Klier <chi@chimeric.de> 82b8fdc796SMichael Klier */ 83b8fdc796SMichael Klier function handle_ajax_call(&$event, $params) { 84b8fdc796SMichael Klier if($event->data != 'discussion_preview') return; 85b8fdc796SMichael Klier $event->preventDefault(); 86b8fdc796SMichael Klier $event->stopPropagation(); 87b8fdc796SMichael Klier print p_locale_xhtml('preview'); 88b8fdc796SMichael Klier print '<div class="comment_preview">'; 89b8fdc796SMichael Klier if(!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) { 90b8fdc796SMichael Klier print p_locale_xhtml('denied'); 91b8fdc796SMichael Klier } else { 9287bb4e97SMichael Klier print $this->_render($_REQUEST['comment']); 93b8fdc796SMichael Klier } 94b8fdc796SMichael Klier print '</div>'; 952d4bee9aSMichael Klier } 962d4bee9aSMichael Klier 972d4bee9aSMichael Klier /** 982d4bee9aSMichael Klier * Modify Tollbar for use with discussion plugin 992d4bee9aSMichael Klier * 1002d4bee9aSMichael Klier * @author Michael Klier <chi@chimeric.de> 1012d4bee9aSMichael Klier */ 1022d4bee9aSMichael Klier function handle_toolbar_define(&$event, $param) { 1032d4bee9aSMichael Klier global $ACT; 1042d4bee9aSMichael Klier if($ACT != 'show') return; 1052d4bee9aSMichael Klier 1061de52da1SMichael Klier if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) { 1072d4bee9aSMichael Klier $toolbar = array(); 1082d4bee9aSMichael Klier foreach($event->data as $btn) { 1092d4bee9aSMichael Klier if($btn['type'] == 'mediapopup') continue; 1102d4bee9aSMichael Klier if($btn['type'] == 'signature') continue; 1112d4bee9aSMichael Klier if(preg_match("/=+?/", $btn['open'])) continue; 1122d4bee9aSMichael Klier array_push($toolbar, $btn); 1132d4bee9aSMichael Klier } 1142d4bee9aSMichael Klier $event->data = $toolbar; 1152d4bee9aSMichael Klier } 1162d4bee9aSMichael Klier } 1172d4bee9aSMichael Klier 1182d4bee9aSMichael Klier /** 1192d4bee9aSMichael Klier * Dirty workaround to add a toolbar to the discussion plugin 1202d4bee9aSMichael Klier * 1212d4bee9aSMichael Klier * @author Michael Klier <chi@chimeric.de> 1222d4bee9aSMichael Klier */ 1232d4bee9aSMichael Klier function handle_tpl_metaheader_output(&$event, $param) { 1242d4bee9aSMichael Klier global $ACT; 1252d4bee9aSMichael Klier global $ID; 1262d4bee9aSMichael Klier if($ACT != 'show') return; 1272d4bee9aSMichael Klier 1282d4bee9aSMichael Klier // FIXME check if this works for global discussion/on too 1291de52da1SMichael Klier if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) { 1302d4bee9aSMichael Klier // FIXME ugly workaround, replace this once DW the toolbar code is more flexible 1312d4bee9aSMichael Klier array_unshift($event->data['script'], array('type' => 'text/javascript', 'charset' => 'utf-8', '_data' => '', 'src' => DOKU_BASE.'lib/scripts/edit.js')); 1322d4bee9aSMichael Klier @require_once(DOKU_INC.'inc/toolbar.php'); 1332d4bee9aSMichael Klier ob_start(); 1342d4bee9aSMichael Klier print 'NS = "' . getNS($ID) . '";'; // we have to define NS, otherwise we get get JS errors 1352d4bee9aSMichael Klier toolbar_JSdefines('toolbar'); 1362d4bee9aSMichael Klier $script = ob_get_clean(); 1372d4bee9aSMichael Klier array_push($event->data['script'], array('type' => 'text/javascript', 'charset' => "utf-8", '_data' => $script)); 1382d4bee9aSMichael Klier } 139f0fda08aSwikidesign } 140f0fda08aSwikidesign 141f0fda08aSwikidesign /** 142a1d93126SGina Haeussge * Handles comment actions, dispatches data processing routines 143f0fda08aSwikidesign */ 144a1d93126SGina Haeussge function handle_act_preprocess(&$event, $param) { 145284f9aa2SMichael Klier global $ID; 146d7eb2e24SMichael Klier global $INFO; 147284f9aa2SMichael Klier global $conf; 1483011fb8bSMichael Klier global $lang; 149573e23a1Swikidesign 150a1d93126SGina Haeussge // handle newthread ACTs 151a1d93126SGina Haeussge if ($event->data == 'newthread') { 152a1d93126SGina Haeussge // we can handle it -> prevent others 153a1d93126SGina Haeussge $event->preventDefault(); 154a1d93126SGina Haeussge $event->data = $this->_newThread(); 155a1d93126SGina Haeussge } 156a1d93126SGina Haeussge 157a1d93126SGina Haeussge // enable captchas 158a1d93126SGina Haeussge if ((in_array($_REQUEST['comment'], array('add', 'save'))) 159a1d93126SGina Haeussge && (@file_exists(DOKU_PLUGIN.'captcha/action.php'))) { 160a1d93126SGina Haeussge $this->_captchaCheck(); 161a1d93126SGina Haeussge } 162a1d93126SGina Haeussge 1633011fb8bSMichael Klier // if we are not in show mode or someone wants to unsubscribe, that was all for now 1649881d835SMichael Klier if ($event->data != 'show' && $event->data != 'unsubscribe' && $event->data != 'confirmsubscribe') return; 165a1d93126SGina Haeussge 1669881d835SMichael Klier if ($event->data == 'unsubscribe' or $event->data == 'confirmsubscribe') { 1679881d835SMichael Klier // ok we can handle it prevent others 1689881d835SMichael Klier $event->preventDefault(); 1699881d835SMichael Klier 1703011fb8bSMichael Klier if (!isset($_REQUEST['hash'])) { 1719881d835SMichael Klier return false; 1723011fb8bSMichael Klier } else { 1733011fb8bSMichael Klier $file = metaFN($ID, '.comments'); 1743011fb8bSMichael Klier $data = unserialize(io_readFile($file)); 1759881d835SMichael Klier foreach($data['subscribers'] as $mail => $info) { 1769881d835SMichael Klier // convert old style subscribers just in case 1779881d835SMichael Klier if(!is_array($info)) { 1789881d835SMichael Klier $hash = $data['subscribers'][$mail]; 1799881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 1809881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 1819881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 1829881d835SMichael Klier } 1839881d835SMichael Klier } 1849881d835SMichael Klier 1859881d835SMichael Klier if($data['subscribers'][$mail]['hash'] == $_REQUEST['hash']) { 1869881d835SMichael Klier if($event->data == 'unsubscribe') { 1873011fb8bSMichael Klier unset($data['subscribers'][$mail]); 1883011fb8bSMichael Klier msg(sprintf($lang['unsubscribe_success'], $mail, $ID), 1); 1899881d835SMichael Klier } elseif($event->data == 'confirmsubscribe') { 1909881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 1919881d835SMichael Klier msg(sprintf($lang['subscribe_success'], $mail, $ID), 1); 1929881d835SMichael Klier } 1939881d835SMichael Klier io_saveFile($file, serialize($data)); 1943011fb8bSMichael Klier $event->data = 'show'; 1953011fb8bSMichael Klier return true; 1969881d835SMichael Klier } else { 1973011fb8bSMichael Klier return false; 1983011fb8bSMichael Klier } 1999881d835SMichael Klier } 2003011fb8bSMichael Klier } else { 201a1d93126SGina Haeussge // do the data processing for comments 202f0fda08aSwikidesign $cid = $_REQUEST['cid']; 203f0fda08aSwikidesign switch ($_REQUEST['comment']) { 204f0fda08aSwikidesign case 'add': 20537e3c825SMichael Klier if(empty($_REQUEST['text'])) return; // don't add empty comments 20694c5d164SMichael Klier if(isset($_SERVER['REMOTE_USER']) && !$this->getConf('adminimport')) { 20794c5d164SMichael Klier $comment['user']['id'] = $_SERVER['REMOTE_USER']; 20894c5d164SMichael Klier $comment['user']['name'] = $INFO['userinfo']['name']; 20994c5d164SMichael Klier $comment['user']['mail'] = $INFO['userinfo']['mail']; 21094c5d164SMichael Klier } elseif((isset($_SERVER['REMOTE_USER']) && $this->getConf('adminimport') && auth_ismanager()) || !isset($_SERVER['REMOTE_USER'])) { 21137e3c825SMichael Klier if(empty($_REQUEST['name']) or empty($_REQUEST['mail'])) return // don't add anonymous comments 21294c5d164SMichael Klier $comment['user']['id'] = 'test'.hsc($_REQUEST['user']); 21394c5d164SMichael Klier $comment['user']['name'] = hsc($_REQUEST['name']); 21494c5d164SMichael Klier $comment['user']['mail'] = hsc($_REQUEST['mail']); 21594c5d164SMichael Klier } 21694c5d164SMichael Klier $comment['user']['address'] = ($this->getConf('addressfield')) ? hsc($_REQUEST['address']) : ''; 217a10b5c98SMichael Klier $comment['user']['url'] = ($this->getConf('urlfield')) ? $this->_checkURL($_REQUEST['url']) : ''; 21894c5d164SMichael Klier $comment['subscribe'] = ($this->getConf('subscribe')) ? $_REQUEST['subscribe'] : ''; 21994c5d164SMichael Klier $comment['date'] = array('created' => $_REQUEST['date']); 22094c5d164SMichael Klier $comment['raw'] = cleanText($_REQUEST['text']); 221f0fda08aSwikidesign $repl = $_REQUEST['reply']; 222b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 223a44bc9f7SMichael Klier $comment['show'] = false; 224a44bc9f7SMichael Klier } else { 225a44bc9f7SMichael Klier $comment['show'] = true; 226a44bc9f7SMichael Klier } 227f0fda08aSwikidesign $this->_add($comment, $repl); 228f0fda08aSwikidesign break; 229f0fda08aSwikidesign 230f0fda08aSwikidesign case 'save': 231f0fda08aSwikidesign $raw = cleanText($_REQUEST['text']); 232264b7327Swikidesign $this->_save(array($cid), $raw); 233f0fda08aSwikidesign break; 234f0fda08aSwikidesign 2351e46d176Swikidesign case 'delete': 236264b7327Swikidesign $this->_save(array($cid), ''); 2372ee3dca3Swikidesign break; 2381e46d176Swikidesign 239f0fda08aSwikidesign case 'toogle': 240264b7327Swikidesign $this->_save(array($cid), '', 'toogle'); 241f0fda08aSwikidesign break; 242a1d93126SGina Haeussge } 2433011fb8bSMichael Klier } 244284f9aa2SMichael Klier 245284f9aa2SMichael Klier // FIXME use new TPL_TOC_RENDER event in the future 246d7eb2e24SMichael Klier if(count($INFO['meta']['description']['tableofcontents']) >= ($conf['maxtoclevel']-1) && $INFO['meta']['internal']['toc']) { 247284f9aa2SMichael Klier 248284f9aa2SMichael Klier $TOC = array(); 249284f9aa2SMichael Klier global $TOC; 250d7eb2e24SMichael Klier $TOC = $INFO['meta']['description']['tableofcontents']; 251284f9aa2SMichael Klier 252284f9aa2SMichael Klier $tocitem = array( 'hid' => 'discussion__section', 253284f9aa2SMichael Klier 'title' => $this->getLang('discussion'), 254284f9aa2SMichael Klier 'type' => 'ul', 255284f9aa2SMichael Klier 'level' => 1 ); 256284f9aa2SMichael Klier 257284f9aa2SMichael Klier $file = metaFN($ID, '.comments'); 258284f9aa2SMichael Klier if(@file_exists($file)) { 259284f9aa2SMichael Klier $data = unserialize(io_readFile($file)); 260284f9aa2SMichael Klier if($data['status'] != 0 && !empty($TOC)) { 261284f9aa2SMichael Klier $TOC[] = $tocitem; 262284f9aa2SMichael Klier } 263284f9aa2SMichael Klier } 264284f9aa2SMichael Klier } 265a1d93126SGina Haeussge } 266a1d93126SGina Haeussge 267a1d93126SGina Haeussge /** 268a1d93126SGina Haeussge * Main function; dispatches the visual comment actions 269a1d93126SGina Haeussge */ 270a1d93126SGina Haeussge function comments(&$event, $param) { 271a1d93126SGina Haeussge if ($event->data != 'show') return; // nothing to do for us 272a1d93126SGina Haeussge 273a1d93126SGina Haeussge $cid = $_REQUEST['cid']; 274a1d93126SGina Haeussge switch ($_REQUEST['comment']) { 275a1d93126SGina Haeussge case 'edit': 276a1d93126SGina Haeussge $this->_show(NULL, $cid); 277a1d93126SGina Haeussge break; 2782b18adb9SMichael Klier default: 279f0fda08aSwikidesign $this->_show($cid); 2802b18adb9SMichael Klier break; 281f0fda08aSwikidesign } 282f0fda08aSwikidesign } 283f0fda08aSwikidesign 284f0fda08aSwikidesign /** 285a1d93126SGina Haeussge * Redirects browser to given comment anchor 286a1d93126SGina Haeussge */ 287a1d93126SGina Haeussge function _redirect($cid) { 288a1d93126SGina Haeussge global $ID; 289a1d93126SGina Haeussge global $ACT; 290a1d93126SGina Haeussge 291a1d93126SGina Haeussge if ($ACT !== 'show') return; 292a44bc9f7SMichael Klier 293b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 294a44bc9f7SMichael Klier msg($this->getLang('moderation'), 1); 295a44bc9f7SMichael Klier @session_start(); 296a44bc9f7SMichael Klier global $MSG; 297a44bc9f7SMichael Klier $_SESSION[DOKU_COOKIE]['msg'] = $MSG; 298a44bc9f7SMichael Klier session_write_close(); 299a44bc9f7SMichael Klier $url = wl($ID); 300a44bc9f7SMichael Klier } else { 301a44bc9f7SMichael Klier $url = wl($ID) . '#comment_' . $cid; 302a44bc9f7SMichael Klier } 303a44bc9f7SMichael Klier send_redirect($url); 3046d1f4f20SGina Haeussge exit(); 305a1d93126SGina Haeussge } 306a1d93126SGina Haeussge 307a1d93126SGina Haeussge /** 308f0fda08aSwikidesign * Shows all comments of the current page 309f0fda08aSwikidesign */ 310f0fda08aSwikidesign function _show($reply = NULL, $edit = NULL) { 3112b18adb9SMichael Klier global $ID; 3122b18adb9SMichael Klier global $INFO; 3132b18adb9SMichael Klier global $ACT; 314573e23a1Swikidesign 315479dd10fSwikidesign // get .comments meta file name 316f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 317f0fda08aSwikidesign 3184aeddd8eSMichael Klier if (!$INFO['exists']) return; 3192b18adb9SMichael Klier if (!@file_exists($file) && !$this->getConf('automatic')) return false; 320f0fda08aSwikidesign 3212b18adb9SMichael Klier // load data 3222b18adb9SMichael Klier if (@file_exists($file)) { 3232b18adb9SMichael Klier $data = unserialize(io_readFile($file, false)); 324479dd10fSwikidesign if (!$data['status']) return false; // comments are turned off 3252b18adb9SMichael Klier } elseif (!@file_exists($file) && $this->getConf('automatic') && $INFO['exists']) { 3262b18adb9SMichael Klier // set status to show the comment form 3272b18adb9SMichael Klier $data['status'] = 1; 3282b18adb9SMichael Klier $data['number'] = 0; 3292b18adb9SMichael Klier } 330f0fda08aSwikidesign 331f0fda08aSwikidesign // section title 33207c376bbSwikidesign $title = ($data['title'] ? hsc($data['title']) : $this->getLang('discussion')); 3334a0a1bd2Swikidesign ptln('<div class="comment_wrapper">'); 3344a0a1bd2Swikidesign ptln('<h2><a name="discussion__section" id="discussion__section">', 2); 3354a0a1bd2Swikidesign ptln($title, 4); 3364a0a1bd2Swikidesign ptln('</a></h2>', 2); 3374a0a1bd2Swikidesign ptln('<div class="level2 hfeed">', 2); 338f0fda08aSwikidesign // now display the comments 339f0fda08aSwikidesign if (isset($data['comments'])) { 34031aab30eSGina Haeussge if (!$this->getConf('usethreading')) { 34131aab30eSGina Haeussge $data['comments'] = $this->_flattenThreads($data['comments']); 34231aab30eSGina Haeussge uasort($data['comments'], '_sortCallBack'); 34331aab30eSGina Haeussge } 344dbd9d5cdSMichael Klier if($this->getConf('newestfirst')) { 345dbd9d5cdSMichael Klier $data['comments'] = array_reverse($data['comments']); 346dbd9d5cdSMichael Klier } 347f0fda08aSwikidesign foreach ($data['comments'] as $key => $value) { 348f0fda08aSwikidesign if ($key == $edit) $this->_form($value['raw'], 'save', $edit); // edit form 349f0fda08aSwikidesign else $this->_print($key, $data, '', $reply); 350f0fda08aSwikidesign } 351f0fda08aSwikidesign } 352f0fda08aSwikidesign 353f0fda08aSwikidesign // comment form 35431aab30eSGina Haeussge if (($data['status'] == 1) && (!$reply || !$this->getConf('usethreading')) && !$edit) $this->_form(''); 355f0fda08aSwikidesign 3564a0a1bd2Swikidesign ptln('</div>', 2); // level2 hfeed 3574a0a1bd2Swikidesign ptln('</div>'); // comment_wrapper 358f0fda08aSwikidesign 359f0fda08aSwikidesign return true; 360f0fda08aSwikidesign } 361f0fda08aSwikidesign 36231aab30eSGina Haeussge function _flattenThreads($comments, $keys = null) { 36331aab30eSGina Haeussge if (is_null($keys)) 36431aab30eSGina Haeussge $keys = array_keys($comments); 36531aab30eSGina Haeussge 36631aab30eSGina Haeussge foreach($keys as $cid) { 36731aab30eSGina Haeussge if (!empty($comments[$cid]['replies'])) { 36831aab30eSGina Haeussge $rids = $comments[$cid]['replies']; 36931aab30eSGina Haeussge $comments = $this->_flattenThreads($comments, $rids); 37031aab30eSGina Haeussge $comments[$cid]['replies'] = array(); 37131aab30eSGina Haeussge } 37231aab30eSGina Haeussge $comments[$cid]['parent'] = ''; 37331aab30eSGina Haeussge } 37431aab30eSGina Haeussge return $comments; 37531aab30eSGina Haeussge } 37631aab30eSGina Haeussge 377f0fda08aSwikidesign /** 378f0fda08aSwikidesign * Adds a new comment and then displays all comments 379f0fda08aSwikidesign */ 380f0fda08aSwikidesign function _add($comment, $parent) { 3813011fb8bSMichael Klier global $lang; 3823011fb8bSMichael Klier global $ID; 3833011fb8bSMichael Klier global $TEXT; 384f0fda08aSwikidesign 385f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 386f0fda08aSwikidesign $TEXT = $comment['raw']; 387f0fda08aSwikidesign 388f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 389f0fda08aSwikidesign if (checkwordblock()) { 390f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 391f0fda08aSwikidesign return false; 392f0fda08aSwikidesign } 393f0fda08aSwikidesign 3943011fb8bSMichael Klier if ((!$this->getConf('allowguests')) 3953011fb8bSMichael Klier && ($comment['user']['id'] != $_SERVER['REMOTE_USER'])) 3963011fb8bSMichael Klier return false; // guest comments not allowed 3973011fb8bSMichael Klier 398f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 399f0fda08aSwikidesign 400f0fda08aSwikidesign // get discussion meta file name 401f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 402f0fda08aSwikidesign 4032b18adb9SMichael Klier // create comments file if it doesn't exist yet 4042b18adb9SMichael Klier if(!@file_exists($file)) { 4052b18adb9SMichael Klier $data = array('status' => 1, 'number' => 0); 4062b18adb9SMichael Klier io_saveFile($file, serialize($data)); 4072b18adb9SMichael Klier } else { 408f0fda08aSwikidesign $data = array(); 409f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 410f0fda08aSwikidesign if ($data['status'] != 1) return false; // comments off or closed 4112b18adb9SMichael Klier } 4122b18adb9SMichael Klier 4133011fb8bSMichael Klier if ($comment['date']['created']) { 4143011fb8bSMichael Klier $date = strtotime($comment['date']['created']); 4153011fb8bSMichael Klier } else { 4163011fb8bSMichael Klier $date = time(); 4173011fb8bSMichael Klier } 418f0fda08aSwikidesign 4193011fb8bSMichael Klier if ($date == -1) { 4203011fb8bSMichael Klier $date = time(); 4213011fb8bSMichael Klier } 4223011fb8bSMichael Klier 4236046f25cSwikidesign $cid = md5($comment['user']['id'].$date); // create a unique id 424f0fda08aSwikidesign 4253011fb8bSMichael Klier if (!is_array($data['comments'][$parent])) { 4263011fb8bSMichael Klier $parent = NULL; // invalid parent comment 4273011fb8bSMichael Klier } 428f0fda08aSwikidesign 429f0fda08aSwikidesign // render the comment 430f0fda08aSwikidesign $xhtml = $this->_render($comment['raw']); 431f0fda08aSwikidesign 432f0fda08aSwikidesign // fill in the new comment 433f0fda08aSwikidesign $data['comments'][$cid] = array( 4346046f25cSwikidesign 'user' => $comment['user'], 4356046f25cSwikidesign 'date' => array('created' => $date), 436f0fda08aSwikidesign 'show' => true, 4376046f25cSwikidesign 'raw' => $comment['raw'], 438f0fda08aSwikidesign 'xhtml' => $xhtml, 439f0fda08aSwikidesign 'parent' => $parent, 440a44bc9f7SMichael Klier 'replies' => array(), 441a44bc9f7SMichael Klier 'show' => $comment['show'] 442f0fda08aSwikidesign ); 443f0fda08aSwikidesign 4443011fb8bSMichael Klier if($comment['subscribe']) { 4453011fb8bSMichael Klier $mail = $comment['user']['mail']; 4463011fb8bSMichael Klier if($data['subscribers']) { 4473011fb8bSMichael Klier if(!$data['subscribers'][$mail]) { 4489881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4499881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4509881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4519881d835SMichael Klier } else { 4529881d835SMichael Klier // convert old style subscribers and set them active 4539881d835SMichael Klier if(!is_array($data['subscribers'][$mail])) { 4549881d835SMichael Klier $hash = $data['subscribers'][$mail]; 4559881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 4569881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 4579881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 4589881d835SMichael Klier } 4593011fb8bSMichael Klier } 4603011fb8bSMichael Klier } else { 4619881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4629881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4639881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4643011fb8bSMichael Klier } 4653011fb8bSMichael Klier } 4663011fb8bSMichael Klier 467f0fda08aSwikidesign // update parent comment 468f0fda08aSwikidesign if ($parent) $data['comments'][$parent]['replies'][] = $cid; 469f0fda08aSwikidesign 470f0fda08aSwikidesign // update the number of comments 471f0fda08aSwikidesign $data['number']++; 472f0fda08aSwikidesign 473f0fda08aSwikidesign // notify subscribers of the page 4748b42cefbSMichael Klier $data['comments'][$cid]['cid'] = $cid; 4753011fb8bSMichael Klier $this->_notify($data['comments'][$cid], $data['subscribers']); 476f0fda08aSwikidesign 4779881d835SMichael Klier // save the comment metadata file 4789881d835SMichael Klier io_saveFile($file, serialize($data)); 4799881d835SMichael Klier $this->_addLogEntry($date, $ID, 'cc', '', $cid); 4809881d835SMichael Klier 481a1d93126SGina Haeussge $this->_redirect($cid); 482f0fda08aSwikidesign return true; 483f0fda08aSwikidesign } 484f0fda08aSwikidesign 485f0fda08aSwikidesign /** 486f0fda08aSwikidesign * Saves the comment with the given ID and then displays all comments 487f0fda08aSwikidesign */ 488264b7327Swikidesign function _save($cids, $raw, $act = NULL) { 489a1ace3c9Swikidesign global $ID; 490f0fda08aSwikidesign 4912ee3dca3Swikidesign if ($raw) { 4922ee3dca3Swikidesign global $TEXT; 4932ee3dca3Swikidesign 494f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 495f0fda08aSwikidesign $TEXT = $raw; 496f0fda08aSwikidesign 497f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 498f0fda08aSwikidesign if (checkwordblock()) { 499f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 500f0fda08aSwikidesign return false; 501f0fda08aSwikidesign } 502f0fda08aSwikidesign 503f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 5042ee3dca3Swikidesign } 505f0fda08aSwikidesign 506f0fda08aSwikidesign // get discussion meta file name 507f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 508f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 509f0fda08aSwikidesign 510573e23a1Swikidesign if (!is_array($cids)) $cids = array($cids); 511264b7327Swikidesign foreach ($cids as $cid) { 512264b7327Swikidesign 5136046f25cSwikidesign if (is_array($data['comments'][$cid]['user'])) { 5146046f25cSwikidesign $user = $data['comments'][$cid]['user']['id']; 5156046f25cSwikidesign $convert = false; 5166046f25cSwikidesign } else { 5176046f25cSwikidesign $user = $data['comments'][$cid]['user']; 5186046f25cSwikidesign $convert = true; 5196046f25cSwikidesign } 5206046f25cSwikidesign 521f0fda08aSwikidesign // someone else was trying to edit our comment -> abort 5223e02b3ffSwikidesign if (($user != $_SERVER['REMOTE_USER']) && (!auth_ismanager())) return false; 523f0fda08aSwikidesign 524f0fda08aSwikidesign $date = time(); 525f0fda08aSwikidesign 5266046f25cSwikidesign // need to convert to new format? 5276046f25cSwikidesign if ($convert) { 5286046f25cSwikidesign $data['comments'][$cid]['user'] = array( 5296046f25cSwikidesign 'id' => $user, 5306046f25cSwikidesign 'name' => $data['comments'][$cid]['name'], 5316046f25cSwikidesign 'mail' => $data['comments'][$cid]['mail'], 5326046f25cSwikidesign 'url' => $data['comments'][$cid]['url'], 5336046f25cSwikidesign 'address' => $data['comments'][$cid]['address'], 5346046f25cSwikidesign ); 5356046f25cSwikidesign $data['comments'][$cid]['date'] = array( 5366046f25cSwikidesign 'created' => $data['comments'][$cid]['date'] 5376046f25cSwikidesign ); 5386046f25cSwikidesign } 5396046f25cSwikidesign 540264b7327Swikidesign if ($act == 'toogle') { // toogle visibility 541f0fda08aSwikidesign $now = $data['comments'][$cid]['show']; 542f0fda08aSwikidesign $data['comments'][$cid]['show'] = !$now; 543f0fda08aSwikidesign $data['number'] = $this->_count($data); 544f0fda08aSwikidesign 545f0fda08aSwikidesign $type = ($data['comments'][$cid]['show'] ? 'sc' : 'hc'); 546f0fda08aSwikidesign 547264b7327Swikidesign } elseif ($act == 'show') { // show comment 548264b7327Swikidesign $data['comments'][$cid]['show'] = true; 549264b7327Swikidesign $data['number'] = $this->_count($data); 550264b7327Swikidesign 551573e23a1Swikidesign $type = 'sc'; // show comment 552264b7327Swikidesign 553264b7327Swikidesign } elseif ($act == 'hide') { // hide comment 554264b7327Swikidesign $data['comments'][$cid]['show'] = false; 555264b7327Swikidesign $data['number'] = $this->_count($data); 556264b7327Swikidesign 557573e23a1Swikidesign $type = 'hc'; // hide comment 558264b7327Swikidesign 559f0fda08aSwikidesign } elseif (!$raw) { // remove the comment 560efbe59d0Swikidesign $data['comments'] = $this->_removeComment($cid, $data['comments']); 561f0fda08aSwikidesign $data['number'] = $this->_count($data); 562f0fda08aSwikidesign 563573e23a1Swikidesign $type = 'dc'; // delete comment 564f0fda08aSwikidesign 565f0fda08aSwikidesign } else { // save changed comment 566f0fda08aSwikidesign $xhtml = $this->_render($raw); 567f0fda08aSwikidesign 568f0fda08aSwikidesign // now change the comment's content 5696046f25cSwikidesign $data['comments'][$cid]['date']['modified'] = $date; 5706046f25cSwikidesign $data['comments'][$cid]['raw'] = $raw; 571f0fda08aSwikidesign $data['comments'][$cid]['xhtml'] = $xhtml; 572f0fda08aSwikidesign 573573e23a1Swikidesign $type = 'ec'; // edit comment 574f0fda08aSwikidesign } 575264b7327Swikidesign } 576264b7327Swikidesign 577f0fda08aSwikidesign // save the comment metadata file 578f0fda08aSwikidesign io_saveFile($file, serialize($data)); 579f0fda08aSwikidesign $this->_addLogEntry($date, $ID, $type, '', $cid); 580f0fda08aSwikidesign 581a1d93126SGina Haeussge $this->_redirect($cid); 582f0fda08aSwikidesign return true; 583f0fda08aSwikidesign } 584f0fda08aSwikidesign 585f0fda08aSwikidesign /** 586efbe59d0Swikidesign * Recursive function to remove a comment 587efbe59d0Swikidesign */ 588efbe59d0Swikidesign function _removeComment($cid, $comments) { 589efbe59d0Swikidesign if (is_array($comments[$cid]['replies'])) { 590efbe59d0Swikidesign foreach ($comments[$cid]['replies'] as $rid) { 591efbe59d0Swikidesign $comments = $this->_removeComment($rid, $comments); 592efbe59d0Swikidesign } 593efbe59d0Swikidesign } 594efbe59d0Swikidesign unset($comments[$cid]); 595efbe59d0Swikidesign return $comments; 596efbe59d0Swikidesign } 597efbe59d0Swikidesign 598efbe59d0Swikidesign /** 599f0fda08aSwikidesign * Prints an individual comment 600f0fda08aSwikidesign */ 601f0fda08aSwikidesign function _print($cid, &$data, $parent = '', $reply = '', $visible = true) { 602f0fda08aSwikidesign 6032ee3dca3Swikidesign if (!isset($data['comments'][$cid])) return false; // comment was removed 604f0fda08aSwikidesign $comment = $data['comments'][$cid]; 605f0fda08aSwikidesign 606f0fda08aSwikidesign if (!is_array($comment)) return false; // corrupt datatype 607f0fda08aSwikidesign 608f0fda08aSwikidesign if ($comment['parent'] != $parent) return true; // reply to an other comment 609f0fda08aSwikidesign 610f0fda08aSwikidesign if (!$comment['show']) { // comment hidden 611a1ace3c9Swikidesign if (auth_ismanager()) $hidden = ' comment_hidden'; 612f0fda08aSwikidesign else return true; 6134a0a1bd2Swikidesign } else { 6144a0a1bd2Swikidesign $hidden = ''; 615f0fda08aSwikidesign } 616f0fda08aSwikidesign 617f1c6610eSpierre.spring if($this->getConf('newestfirst')) { 618f1c6610eSpierre.spring // reply form 619f1c6610eSpierre.spring $this->_print_form($cid, $reply); 620f1c6610eSpierre.spring // replies to this comment entry? 621f1c6610eSpierre.spring $this->_print_replies($cid, $data, $reply, $visible); 622f1c6610eSpierre.spring // print the actual comment 6236b1c1263SMichael Klier $this->_print_comment($cid, $data, $parent, $reply, $visible, $hidden); 624f1c6610eSpierre.spring } else { 625f1c6610eSpierre.spring // print the actual comment 6266b1c1263SMichael Klier $this->_print_comment($cid, $data, $parent, $reply, $visible, $hidden); 627f1c6610eSpierre.spring // replies to this comment entry? 628f1c6610eSpierre.spring $this->_print_replies($cid, $data, $reply, $visible); 629f1c6610eSpierre.spring // reply form 630f1c6610eSpierre.spring $this->_print_form($cid, $reply); 631f1c6610eSpierre.spring } 632f1c6610eSpierre.spring } 633f1c6610eSpierre.spring 634f1c6610eSpierre.spring function _print_comment($cid, &$data, $parent, $reply, $visible, $hidden) 635f1c6610eSpierre.spring { 636f1c6610eSpierre.spring global $conf, $lang, $ID, $HIGH; 637f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 638f1c6610eSpierre.spring 639f0fda08aSwikidesign // comment head with date and user data 6404a0a1bd2Swikidesign ptln('<div class="hentry'.$hidden.'">', 4); 6414a0a1bd2Swikidesign ptln('<div class="comment_head">', 6); 6421810ba9cSMichael Klier ptln('<a name="comment_'.$cid.'" id="comment_'.$cid.'"></a>', 8); 6434a0a1bd2Swikidesign $head = '<span class="vcard author">'; 644f0fda08aSwikidesign 6456046f25cSwikidesign // prepare variables 6466046f25cSwikidesign if (is_array($comment['user'])) { // new format 6476046f25cSwikidesign $user = $comment['user']['id']; 6486046f25cSwikidesign $name = $comment['user']['name']; 6496046f25cSwikidesign $mail = $comment['user']['mail']; 6506046f25cSwikidesign $url = $comment['user']['url']; 6516046f25cSwikidesign $address = $comment['user']['address']; 6526046f25cSwikidesign } else { // old format 6536046f25cSwikidesign $user = $comment['user']; 6546046f25cSwikidesign $name = $comment['name']; 6556046f25cSwikidesign $mail = $comment['mail']; 6566046f25cSwikidesign $url = $comment['url']; 6576046f25cSwikidesign $address = $comment['address']; 6586046f25cSwikidesign } 6596046f25cSwikidesign if (is_array($comment['date'])) { // new format 6606046f25cSwikidesign $created = $comment['date']['created']; 6616046f25cSwikidesign $modified = $comment['date']['modified']; 6626046f25cSwikidesign } else { // old format 6636046f25cSwikidesign $created = $comment['date']; 6646046f25cSwikidesign $modified = $comment['edited']; 6656046f25cSwikidesign } 6666046f25cSwikidesign 667ce7b17cfSwikidesign // show avatar image? 668f1c6610eSpierre.spring if ($this->_use_avatar()) { 669*db963b87SMichael Klier if(!$mail) $mail = $name; 67011fb5dc5SMichael Klier $avatar = $this->avatar->getXHTML($mail, $name, 'left'); 671*db963b87SMichael Klier if($avatar) $head .= $avatar; 672f0fda08aSwikidesign } 673f0fda08aSwikidesign 6746046f25cSwikidesign if ($this->getConf('linkemail') && $mail) { 6754a0a1bd2Swikidesign $head .= $this->email($mail, $name, 'email fn'); 6766046f25cSwikidesign } elseif ($url) { 677ab7c9f30SMichael Klier $head .= $this->external_link($this->_checkURL($url), $name, 'urlextern url fn'); 678f0fda08aSwikidesign } else { 6794a0a1bd2Swikidesign $head .= '<span class="fn">'.$name.'</span>'; 680f0fda08aSwikidesign } 6814a0a1bd2Swikidesign if ($address) $head .= ', <span class="adr">'.$address.'</span>'; 6824a0a1bd2Swikidesign $head .= '</span>, '. 683f014bc86SMichael Klier '<abbr class="published" title="'.strftime('%Y-%m-%dT%H:%M:%SZ', $created).'">'. 684f014bc86SMichael Klier strftime($conf['dformat'], $created).'</abbr>'; 6854a0a1bd2Swikidesign if ($comment['edited']) $head .= ' (<abbr class="updated" title="'. 686f014bc86SMichael Klier strftime('%Y-%m-%dT%H:%M:%SZ', $modified).'">'.strftime($conf['dformat'], $modified). 6876046f25cSwikidesign '</abbr>)'; 688f014bc86SMichael Klier ptln($head, 8); 6894a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_head" 690f0fda08aSwikidesign 691f0fda08aSwikidesign // main comment content 6924a0a1bd2Swikidesign ptln('<div class="comment_body entry-content"'. 693f1c6610eSpierre.spring ($this->getConf('useavatar') ? $this->_get_style() : '').'>', 6); 69415cdad37Spierre.spring echo ($HIGH?html_hilight($comment['xhtml'],$HIGH):$comment['xhtml']).DOKU_LF; 6954a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_body" 696f0fda08aSwikidesign 697f0fda08aSwikidesign if ($visible) { 6981184c36aSwikidesign ptln('<div class="comment_buttons">', 6); 699f0fda08aSwikidesign 700f0fda08aSwikidesign // show reply button? 701f1c4aa1aSwikidesign if (($data['status'] == 1) && !$reply && $comment['show'] 70231aab30eSGina Haeussge && ($this->getConf('allowguests') || $_SERVER['REMOTE_USER']) && $this->getConf('usethreading')) 7031e46d176Swikidesign $this->_button($cid, $this->getLang('btn_reply'), 'reply', true); 704f0fda08aSwikidesign 7051184c36aSwikidesign // show edit, show/hide and delete button? 7061184c36aSwikidesign if ((($user == $_SERVER['REMOTE_USER']) && ($user != '')) || (auth_ismanager())) { 7071e46d176Swikidesign $this->_button($cid, $lang['btn_secedit'], 'edit', true); 7081184c36aSwikidesign $label = ($comment['show'] ? $this->getLang('btn_hide') : $this->getLang('btn_show')); 7091184c36aSwikidesign $this->_button($cid, $label, 'toogle'); 7101184c36aSwikidesign $this->_button($cid, $lang['btn_delete'], 'delete'); 711f0fda08aSwikidesign } 7121184c36aSwikidesign ptln('</div>', 6); // class="comment_buttons" 7131184c36aSwikidesign } 7141184c36aSwikidesign ptln('</div>', 4); // class="hentry" 715f0fda08aSwikidesign } 716f0fda08aSwikidesign 717f1c6610eSpierre.spring function _print_form($cid, $reply) 718f1c6610eSpierre.spring { 71931aab30eSGina Haeussge if ($this->getConf('usethreading') && $reply == $cid) { 7204a0a1bd2Swikidesign ptln('<div class="comment_replies">', 4); 721f0fda08aSwikidesign $this->_form('', 'add', $cid); 7224a0a1bd2Swikidesign ptln('</div>', 4); // class="comment_replies" 723f0fda08aSwikidesign } 724f0fda08aSwikidesign } 725f0fda08aSwikidesign 726f1c6610eSpierre.spring function _print_replies($cid, &$data, $reply, &$visible) 727f1c6610eSpierre.spring { 728f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 729f1c6610eSpierre.spring if (!count($comment['replies'])) { 730f1c6610eSpierre.spring return; 731f1c6610eSpierre.spring } 732f1c6610eSpierre.spring ptln('<div class="comment_replies"'.$this->_get_style().'>', 4); 733f1c6610eSpierre.spring $visible = ($comment['show'] && $visible); 734f1c6610eSpierre.spring foreach ($comment['replies'] as $rid) { 735f1c6610eSpierre.spring $this->_print($rid, $data, $cid, $reply, $visible); 736f1c6610eSpierre.spring } 737f1c6610eSpierre.spring ptln('</div>', 4); 738f1c6610eSpierre.spring } 739f1c6610eSpierre.spring 740f1c6610eSpierre.spring function _use_avatar() 741f1c6610eSpierre.spring { 742f1c6610eSpierre.spring if (is_null($this->use_avatar)) { 743f1c6610eSpierre.spring $this->use_avatar = $this->getConf('useavatar') 744f1c6610eSpierre.spring && (!plugin_isdisabled('avatar')) 745f1c6610eSpierre.spring && ($this->avatar =& plugin_load('helper', 'avatar')); 746f1c6610eSpierre.spring } 747f1c6610eSpierre.spring return $this->use_avatar; 748f1c6610eSpierre.spring } 749f1c6610eSpierre.spring 750f1c6610eSpierre.spring function _get_style() 751f1c6610eSpierre.spring { 752f1c6610eSpierre.spring if (is_null($this->style)){ 753f1c6610eSpierre.spring if ($this->_use_avatar()) { 754f1c6610eSpierre.spring $this->style = ' style="margin-left: '.($this->avatar->getConf('size') + 14).'px;"'; 755f1c6610eSpierre.spring } else { 756f1c6610eSpierre.spring $this->style = ' style="margin-left: 20px;"'; 757f1c6610eSpierre.spring } 758f1c6610eSpierre.spring } 759f1c6610eSpierre.spring return $this->style; 760f1c6610eSpierre.spring } 761f1c6610eSpierre.spring 762f0fda08aSwikidesign /** 763f0fda08aSwikidesign * Outputs the comment form 764f0fda08aSwikidesign */ 765f0fda08aSwikidesign function _form($raw = '', $act = 'add', $cid = NULL) { 7663011fb8bSMichael Klier global $lang; 7673011fb8bSMichael Klier global $conf; 7683011fb8bSMichael Klier global $ID; 7693011fb8bSMichael Klier global $INFO; 770f0fda08aSwikidesign 771f0fda08aSwikidesign // not for unregistered users when guest comments aren't allowed 772f0fda08aSwikidesign if (!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) return false; 773f0fda08aSwikidesign 7741ba72c23Swikidesign // fill $raw with $_REQUEST['text'] if it's empty (for failed CAPTCHA check) 7751ba72c23Swikidesign if (!$raw && ($_REQUEST['comment'] == 'show')) $raw = $_REQUEST['text']; 776f0fda08aSwikidesign ?> 7775ef1705fSiLoveiDo 778f0fda08aSwikidesign <div class="comment_form"> 779bea69264SMichael Klier <form id="discussion__comment_form" method="post" action="<?php echo script() ?>" accept-charset="<?php echo $lang['encoding'] ?>"> 780f0fda08aSwikidesign <div class="no"> 781f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 78261437513Swikidesign <input type="hidden" name="do" value="show" /> 783f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 784530693fbSMichael Klier <?php 785f0fda08aSwikidesign // for adding a comment 786f0fda08aSwikidesign if ($act == 'add') { 787f0fda08aSwikidesign ?> 788f0fda08aSwikidesign <input type="hidden" name="reply" value="<?php echo $cid ?>" /> 789f0fda08aSwikidesign <?php 79020c152acSMichael Klier // for guest/adminimport: show name, e-mail and subscribe to comments fields 79120c152acSMichael Klier if(!$_SERVER['REMOTE_USER'] or ($this->getConf('adminimport') && auth_ismanager())) { 792f0fda08aSwikidesign ?> 793f0fda08aSwikidesign <input type="hidden" name="user" value="<?php echo clientIP() ?>" /> 794f0fda08aSwikidesign <div class="comment_name"> 795f0fda08aSwikidesign <label class="block" for="discussion__comment_name"> 796f0fda08aSwikidesign <span><?php echo $lang['fullname'] ?>:</span> 79737e3c825SMichael 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'])?>" /> 798f0fda08aSwikidesign </label> 799f0fda08aSwikidesign </div> 800f0fda08aSwikidesign <div class="comment_mail"> 801f0fda08aSwikidesign <label class="block" for="discussion__comment_mail"> 802f0fda08aSwikidesign <span><?php echo $lang['email'] ?>:</span> 80337e3c825SMichael 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'])?>" /> 804f0fda08aSwikidesign </label> 805f0fda08aSwikidesign </div> 806f0fda08aSwikidesign <?php 807f0fda08aSwikidesign } 808f0fda08aSwikidesign 809f0fda08aSwikidesign // allow entering an URL 810f0fda08aSwikidesign if ($this->getConf('urlfield')) { 811f0fda08aSwikidesign ?> 812f0fda08aSwikidesign <div class="comment_url"> 813f0fda08aSwikidesign <label class="block" for="discussion__comment_url"> 814f0fda08aSwikidesign <span><?php echo $this->getLang('url') ?>:</span> 815e7c760b3Swikidesign <input type="text" class="edit" name="url" id="discussion__comment_url" size="50" tabindex="3" value="<?php echo hsc($_REQUEST['url'])?>" /> 816f0fda08aSwikidesign </label> 817f0fda08aSwikidesign </div> 818f0fda08aSwikidesign <?php 819f0fda08aSwikidesign } 820f0fda08aSwikidesign 821f0fda08aSwikidesign // allow entering an address 822f0fda08aSwikidesign if ($this->getConf('addressfield')) { 823f0fda08aSwikidesign ?> 824f0fda08aSwikidesign <div class="comment_address"> 825f0fda08aSwikidesign <label class="block" for="discussion__comment_address"> 826f0fda08aSwikidesign <span><?php echo $this->getLang('address') ?>:</span> 827e7c760b3Swikidesign <input type="text" class="edit" name="address" id="discussion__comment_address" size="50" tabindex="4" value="<?php echo hsc($_REQUEST['address'])?>" /> 828f0fda08aSwikidesign </label> 829f0fda08aSwikidesign </div> 830f0fda08aSwikidesign <?php 831f0fda08aSwikidesign } 832f0fda08aSwikidesign 833f0fda08aSwikidesign // allow setting the comment date 834a1ace3c9Swikidesign if ($this->getConf('adminimport') && (auth_ismanager())) { 835f0fda08aSwikidesign ?> 836f0fda08aSwikidesign <div class="comment_date"> 837f0fda08aSwikidesign <label class="block" for="discussion__comment_date"> 838f0fda08aSwikidesign <span><?php echo $this->getLang('date') ?>:</span> 839f0fda08aSwikidesign <input type="text" class="edit" name="date" id="discussion__comment_date" size="50" /> 840f0fda08aSwikidesign </label> 841f0fda08aSwikidesign </div> 842f0fda08aSwikidesign <?php 843f0fda08aSwikidesign } 844f0fda08aSwikidesign 845f0fda08aSwikidesign // for saving a comment 846f0fda08aSwikidesign } else { 847f0fda08aSwikidesign ?> 848f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 849f0fda08aSwikidesign <?php 850f0fda08aSwikidesign } 851f0fda08aSwikidesign ?> 852f0fda08aSwikidesign <div class="comment_text"> 8531de52da1SMichael Klier <div id="discussion__comment_toolbar"> 8541de52da1SMichael Klier <?php echo $this->getLang('entercomment')?> 8551de52da1SMichael Klier <?php if($this->getLang('wikisyntaxok')) echo ', ' . $this->getLang('wikisyntax') . ':';?> 8561de52da1SMichael Klier </div> 85737e3c825SMichael 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 85837e3c825SMichael Klier if($raw) { 85937e3c825SMichael Klier echo formText($raw); 86037e3c825SMichael Klier } else { 86137e3c825SMichael Klier echo $_REQUEST['text']; 86237e3c825SMichael Klier } 86337e3c825SMichael Klier ?></textarea> 864f0fda08aSwikidesign </div> 865e7c760b3Swikidesign <?php //bad and dirty event insert hook 866e7c760b3Swikidesign $evdata = array('writable' => true); 867e7c760b3Swikidesign trigger_event('HTML_EDITFORM_INJECTION', $evdata); 868e7c760b3Swikidesign ?> 869bea69264SMichael 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" /> 870ed4846efSMichael Klier <input class="button comment_preview" id="discussion__btn_preview" type="button" name="preview" accesskey="p" value="<?php echo $lang['btn_preview'] ?>" title="<?php echo $lang['btn_preview']?> [P]" /> 8713011fb8bSMichael Klier 872189071d2SMichael Klier <?php if((!$_SERVER['REMOTE_USER'] || $_SERVER['REMOTE_USER'] && !$conf['subscribers']) && $this->getConf('subscribe')) { ?> 8733011fb8bSMichael Klier <div class="comment_subscribe"> 87411b1339eSGina Haeussge <input type="checkbox" id="discussion__comment_subscribe" name="subscribe" tabindex="6" /> 8753011fb8bSMichael Klier <label class="block" for="discussion__comment_subscribe"> 8763011fb8bSMichael Klier <span><?php echo $this->getLang('subscribe') ?></span> 8773011fb8bSMichael Klier </label> 8783011fb8bSMichael Klier </div> 8793011fb8bSMichael Klier <?php } ?> 8803011fb8bSMichael Klier 8813011fb8bSMichael Klier <div class="clearer"></div> 882ed4846efSMichael Klier <div id="discussion__comment_preview"> </div> 883f0fda08aSwikidesign </div> 884f0fda08aSwikidesign </form> 885f0fda08aSwikidesign </div> 886f0fda08aSwikidesign <?php 887f0fda08aSwikidesign if ($this->getConf('usecocomment')) echo $this->_coComment(); 888f0fda08aSwikidesign } 889f0fda08aSwikidesign 890f0fda08aSwikidesign /** 891f0fda08aSwikidesign * Adds a javascript to interact with coComments 892f0fda08aSwikidesign */ 893f0fda08aSwikidesign function _coComment() { 8942b18adb9SMichael Klier global $ID; 8952b18adb9SMichael Klier global $conf; 8962b18adb9SMichael Klier global $INFO; 897f0fda08aSwikidesign 898f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 899f0fda08aSwikidesign 900f0fda08aSwikidesign ?> 901f0fda08aSwikidesign <script type="text/javascript"><!--//--><![CDATA[//><!-- 902f0fda08aSwikidesign var blogTool = "DokuWiki"; 903f0fda08aSwikidesign var blogURL = "<?php echo DOKU_URL ?>"; 904f0fda08aSwikidesign var blogTitle = "<?php echo $conf['title'] ?>"; 905f0fda08aSwikidesign var postURL = "<?php echo wl($ID, '', true) ?>"; 906f0fda08aSwikidesign var postTitle = "<?php echo tpl_pagetitle($ID, true) ?>"; 907f0fda08aSwikidesign <?php 908f0fda08aSwikidesign if ($user) { 909f0fda08aSwikidesign ?> 910f0fda08aSwikidesign var commentAuthor = "<?php echo $INFO['userinfo']['name'] ?>"; 911f0fda08aSwikidesign <?php 912f0fda08aSwikidesign } else { 913f0fda08aSwikidesign ?> 914f0fda08aSwikidesign var commentAuthorFieldName = "name"; 915f0fda08aSwikidesign <?php 916f0fda08aSwikidesign } 917f0fda08aSwikidesign ?> 918f0fda08aSwikidesign var commentAuthorLoggedIn = <?php echo ($user ? 'true' : 'false') ?>; 919f0fda08aSwikidesign var commentFormID = "discussion__comment_form"; 920f0fda08aSwikidesign var commentTextFieldName = "text"; 921f0fda08aSwikidesign var commentButtonName = "submit"; 922f0fda08aSwikidesign var cocomment_force = false; 923f0fda08aSwikidesign //--><!]]></script> 924f0fda08aSwikidesign <script type="text/javascript" src="http://www.cocomment.com/js/cocomment.js"> 925f0fda08aSwikidesign </script> 926f0fda08aSwikidesign <?php 927f0fda08aSwikidesign } 928f0fda08aSwikidesign 929f0fda08aSwikidesign /** 930f0fda08aSwikidesign * General button function 931f0fda08aSwikidesign */ 9321e46d176Swikidesign function _button($cid, $label, $act, $jump = false) { 933f0fda08aSwikidesign global $ID; 9345ef1705fSiLoveiDo 9351e46d176Swikidesign $anchor = ($jump ? '#discussion__comment_form' : '' ); 936f0fda08aSwikidesign 937f0fda08aSwikidesign ?> 9386d1f4f20SGina Haeussge <form class="button discussion__<?php echo $act?>" method="get" action="<?php echo script().$anchor ?>"> 939f0fda08aSwikidesign <div class="no"> 940f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 94161437513Swikidesign <input type="hidden" name="do" value="show" /> 942f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 943f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 944f0fda08aSwikidesign <input type="submit" value="<?php echo $label ?>" class="button" title="<?php echo $label ?>" /> 945f0fda08aSwikidesign </div> 946f0fda08aSwikidesign </form> 947f0fda08aSwikidesign <?php 948f0fda08aSwikidesign return true; 949f0fda08aSwikidesign } 950f0fda08aSwikidesign 951f0fda08aSwikidesign /** 952f0fda08aSwikidesign * Adds an entry to the comments changelog 953f0fda08aSwikidesign * 954f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 955f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 956f0fda08aSwikidesign */ 957f0fda08aSwikidesign function _addLogEntry($date, $id, $type = 'cc', $summary = '', $extra = '') { 958f0fda08aSwikidesign global $conf; 959f0fda08aSwikidesign 960f0fda08aSwikidesign $changelog = $conf['metadir'].'/_comments.changes'; 961f0fda08aSwikidesign 962f0fda08aSwikidesign if(!$date) $date = time(); //use current time if none supplied 963f0fda08aSwikidesign $remote = $_SERVER['REMOTE_ADDR']; 964f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 965f0fda08aSwikidesign 966f0fda08aSwikidesign $strip = array("\t", "\n"); 967f0fda08aSwikidesign $logline = array( 968f0fda08aSwikidesign 'date' => $date, 969f0fda08aSwikidesign 'ip' => $remote, 970f0fda08aSwikidesign 'type' => str_replace($strip, '', $type), 971f0fda08aSwikidesign 'id' => $id, 972f0fda08aSwikidesign 'user' => $user, 973f0fda08aSwikidesign 'sum' => str_replace($strip, '', $summary), 974f0fda08aSwikidesign 'extra' => str_replace($strip, '', $extra) 975f0fda08aSwikidesign ); 976f0fda08aSwikidesign 977f0fda08aSwikidesign // add changelog line 978f0fda08aSwikidesign $logline = implode("\t", $logline)."\n"; 979f0fda08aSwikidesign io_saveFile($changelog, $logline, true); //global changelog cache 980f0fda08aSwikidesign $this->_trimRecentCommentsLog($changelog); 98177a22ba2Swikidesign 98277a22ba2Swikidesign // tell the indexer to re-index the page 98377a22ba2Swikidesign @unlink(metaFN($id, '.indexed')); 984f0fda08aSwikidesign } 985f0fda08aSwikidesign 986f0fda08aSwikidesign /** 987f0fda08aSwikidesign * Trims the recent comments cache to the last $conf['changes_days'] recent 988f0fda08aSwikidesign * changes or $conf['recent'] items, which ever is larger. 989f0fda08aSwikidesign * The trimming is only done once a day. 990f0fda08aSwikidesign * 991f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 992f0fda08aSwikidesign */ 993f0fda08aSwikidesign function _trimRecentCommentsLog($changelog) { 994f0fda08aSwikidesign global $conf; 995f0fda08aSwikidesign 996f0fda08aSwikidesign if (@file_exists($changelog) && 997f0fda08aSwikidesign (filectime($changelog) + 86400) < time() && 998f0fda08aSwikidesign !@file_exists($changelog.'_tmp')) { 999f0fda08aSwikidesign 1000f0fda08aSwikidesign io_lock($changelog); 1001f0fda08aSwikidesign $lines = file($changelog); 1002f0fda08aSwikidesign if (count($lines)<$conf['recent']) { 1003f0fda08aSwikidesign // nothing to trim 1004f0fda08aSwikidesign io_unlock($changelog); 1005f0fda08aSwikidesign return true; 1006f0fda08aSwikidesign } 1007f0fda08aSwikidesign 1008f0fda08aSwikidesign io_saveFile($changelog.'_tmp', ''); // presave tmp as 2nd lock 1009f0fda08aSwikidesign $trim_time = time() - $conf['recent_days']*86400; 1010f0fda08aSwikidesign $out_lines = array(); 1011f0fda08aSwikidesign 1012e49085a2SMichael Klier $num = count($lines); 1013e49085a2SMichael Klier for ($i=0; $i<$num; $i++) { 1014f0fda08aSwikidesign $log = parseChangelogLine($lines[$i]); 1015f0fda08aSwikidesign if ($log === false) continue; // discard junk 1016f0fda08aSwikidesign if ($log['date'] < $trim_time) { 1017f0fda08aSwikidesign $old_lines[$log['date'].".$i"] = $lines[$i]; // keep old lines for now (append .$i to prevent key collisions) 1018f0fda08aSwikidesign } else { 1019f0fda08aSwikidesign $out_lines[$log['date'].".$i"] = $lines[$i]; // definitely keep these lines 1020f0fda08aSwikidesign } 1021f0fda08aSwikidesign } 1022f0fda08aSwikidesign 1023f0fda08aSwikidesign // sort the final result, it shouldn't be necessary, 1024f0fda08aSwikidesign // however the extra robustness in making the changelog cache self-correcting is worth it 1025f0fda08aSwikidesign ksort($out_lines); 1026f0fda08aSwikidesign $extra = $conf['recent'] - count($out_lines); // do we need extra lines do bring us up to minimum 1027f0fda08aSwikidesign if ($extra > 0) { 1028f0fda08aSwikidesign ksort($old_lines); 1029f0fda08aSwikidesign $out_lines = array_merge(array_slice($old_lines,-$extra),$out_lines); 1030f0fda08aSwikidesign } 1031f0fda08aSwikidesign 1032f0fda08aSwikidesign // save trimmed changelog 1033f0fda08aSwikidesign io_saveFile($changelog.'_tmp', implode('', $out_lines)); 1034f0fda08aSwikidesign @unlink($changelog); 1035f0fda08aSwikidesign if (!rename($changelog.'_tmp', $changelog)) { 1036f0fda08aSwikidesign // rename failed so try another way... 1037f0fda08aSwikidesign io_unlock($changelog); 1038f0fda08aSwikidesign io_saveFile($changelog, implode('', $out_lines)); 1039f0fda08aSwikidesign @unlink($changelog.'_tmp'); 1040f0fda08aSwikidesign } else { 1041f0fda08aSwikidesign io_unlock($changelog); 1042f0fda08aSwikidesign } 1043f0fda08aSwikidesign return true; 1044f0fda08aSwikidesign } 1045f0fda08aSwikidesign } 1046f0fda08aSwikidesign 1047f0fda08aSwikidesign /** 1048f0fda08aSwikidesign * Sends a notify mail on new comment 1049f0fda08aSwikidesign * 1050f0fda08aSwikidesign * @param array $comment data array of the new comment 1051f0fda08aSwikidesign * 1052f0fda08aSwikidesign * @author Andreas Gohr <andi@splitbrain.org> 1053f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 1054f0fda08aSwikidesign */ 10559881d835SMichael Klier function _notify($comment, &$subscribers) { 1056f0fda08aSwikidesign global $conf; 1057f0fda08aSwikidesign global $ID; 1058f0fda08aSwikidesign 10599881d835SMichael Klier $notify_text = io_readfile($this->localfn('subscribermail')); 10609881d835SMichael Klier $confirm_text = io_readfile($this->localfn('confirmsubscribe')); 10619881d835SMichael Klier $subject_notify = '['.$conf['title'].'] '.$this->getLang('mail_newcomment'); 10629881d835SMichael Klier $subject_subscribe = '['.$conf['title'].'] '.$this->getLang('subscribe'); 1063f0fda08aSwikidesign 10646046f25cSwikidesign $search = array( 10656046f25cSwikidesign '@PAGE@', 10666046f25cSwikidesign '@TITLE@', 10676046f25cSwikidesign '@DATE@', 10686046f25cSwikidesign '@NAME@', 10696046f25cSwikidesign '@TEXT@', 10708b42cefbSMichael Klier '@COMMENTURL@', 10716046f25cSwikidesign '@UNSUBSCRIBE@', 10726046f25cSwikidesign '@DOKUWIKIURL@', 10736046f25cSwikidesign ); 10743011fb8bSMichael Klier 10753011fb8bSMichael Klier // notify page subscribers 10768c6e87cfSMichael Klier if ($conf['subscribers'] || $conf['notify']) { 1077189071d2SMichael Klier $list = explode(',', subscriber_addresslist($ID)); 10788c6e87cfSMichael Klier $to = (!empty($conf['notify'])) ? $conf['notify'] : array_pop($list); 1079189071d2SMichael Klier $bcc = implode(',', $list); 10803011fb8bSMichael Klier 10816046f25cSwikidesign $replace = array( 10826046f25cSwikidesign $ID, 10836046f25cSwikidesign $conf['title'], 1084f014bc86SMichael Klier strftime($conf['dformat'], $comment['date']['created']), 10856046f25cSwikidesign $comment['user']['name'], 10866046f25cSwikidesign $comment['raw'], 10871810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 10886046f25cSwikidesign wl($ID, 'do=unsubscribe', true, '&'), 10896046f25cSwikidesign DOKU_URL, 10906046f25cSwikidesign ); 1091f0fda08aSwikidesign 10929881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 10939881d835SMichael Klier mail_send($to, $subject_notify, $body, $conf['mailfrom'], '', $bcc); 10943011fb8bSMichael Klier } 1095f0fda08aSwikidesign 10963011fb8bSMichael Klier // notify comment subscribers 10973011fb8bSMichael Klier if (!empty($subscribers)) { 10983011fb8bSMichael Klier 10999881d835SMichael Klier foreach($subscribers as $mail => $data) { 11003011fb8bSMichael Klier $to = $mail; 11013011fb8bSMichael Klier 11029881d835SMichael Klier if($data['active']) { 11033011fb8bSMichael Klier $replace = array( 11043011fb8bSMichael Klier $ID, 11053011fb8bSMichael Klier $conf['title'], 11063011fb8bSMichael Klier strftime($conf['dformat'], $comment['date']['created']), 11073011fb8bSMichael Klier $comment['user']['name'], 11083011fb8bSMichael Klier $comment['raw'], 11091810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 11109881d835SMichael Klier wl($ID, 'do=unsubscribe&hash=' . $data['hash'], true, '&'), 11113011fb8bSMichael Klier DOKU_URL, 11123011fb8bSMichael Klier ); 11133011fb8bSMichael Klier 11149881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 11159881d835SMichael Klier mail_send($to, $subject_notify, $body, $conf['mailfrom']); 11169881d835SMichael Klier } elseif(!$data['active'] && !$data['confirmsent']) { 11179881d835SMichael Klier $search = array( 11189881d835SMichael Klier '@PAGE@', 11199881d835SMichael Klier '@TITLE@', 11209881d835SMichael Klier '@SUBSCRIBE@', 11219881d835SMichael Klier '@DOKUWIKIURL@', 11229881d835SMichael Klier ); 11239881d835SMichael Klier $replace = array( 11249881d835SMichael Klier $ID, 11259881d835SMichael Klier $conf['title'], 11269881d835SMichael Klier wl($ID, 'do=confirmsubscribe&hash=' . $data['hash'], true, '&'), 11279881d835SMichael Klier DOKU_URL, 11289881d835SMichael Klier ); 11299881d835SMichael Klier 11309881d835SMichael Klier $body = str_replace($search, $replace, $confirm_text); 11319881d835SMichael Klier mail_send($to, $subject_subscribe, $body, $conf['mailfrom']); 11329881d835SMichael Klier $subscribers[$mail]['confirmsent'] = true; 11339881d835SMichael Klier } 11343011fb8bSMichael Klier } 11353011fb8bSMichael Klier } 1136f0fda08aSwikidesign } 1137f0fda08aSwikidesign 1138f0fda08aSwikidesign /** 1139f0fda08aSwikidesign * Counts the number of visible comments 1140f0fda08aSwikidesign */ 1141f0fda08aSwikidesign function _count($data) { 1142f0fda08aSwikidesign $number = 0; 1143f0fda08aSwikidesign foreach ($data['comments'] as $cid => $comment) { 1144f0fda08aSwikidesign if ($comment['parent']) continue; 1145f0fda08aSwikidesign if (!$comment['show']) continue; 1146f0fda08aSwikidesign $number++; 1147f0fda08aSwikidesign $rids = $comment['replies']; 1148f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1149f0fda08aSwikidesign } 1150f0fda08aSwikidesign return $number; 1151f0fda08aSwikidesign } 1152f0fda08aSwikidesign 1153f0fda08aSwikidesign function _countReplies(&$data, $rids) { 1154f0fda08aSwikidesign $number = 0; 1155f0fda08aSwikidesign foreach ($rids as $rid) { 11562ee3dca3Swikidesign if (!isset($data['comments'][$rid])) continue; // reply was removed 1157f0fda08aSwikidesign if (!$data['comments'][$rid]['show']) continue; 1158f0fda08aSwikidesign $number++; 1159f0fda08aSwikidesign $rids = $data['comments'][$rid]['replies']; 1160f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1161f0fda08aSwikidesign } 1162f0fda08aSwikidesign return $number; 1163f0fda08aSwikidesign } 1164f0fda08aSwikidesign 1165f0fda08aSwikidesign /** 1166f0fda08aSwikidesign * Renders the comment text 1167f0fda08aSwikidesign */ 1168f0fda08aSwikidesign function _render($raw) { 1169f0fda08aSwikidesign if ($this->getConf('wikisyntaxok')) { 1170f0fda08aSwikidesign $xhtml = $this->render($raw); 1171f0fda08aSwikidesign } else { // wiki syntax not allowed -> just encode special chars 117287bb4e97SMichael Klier $xhtml = hsc(trim($raw)); 117387bb4e97SMichael Klier $xhtml = str_replace("\n", '<br />', $xhtml); 1174f0fda08aSwikidesign } 1175f0fda08aSwikidesign return $xhtml; 1176f0fda08aSwikidesign } 1177f0fda08aSwikidesign 1178f0fda08aSwikidesign /** 1179479dd10fSwikidesign * Finds out whether there is a discussion section for the current page 1180479dd10fSwikidesign */ 1181e6a89be2Swikidesign function _hasDiscussion(&$title) { 1182b2ac3b3bSwikidesign global $ID; 11834a0a1bd2Swikidesign 1184479dd10fSwikidesign $cfile = metaFN($ID, '.comments'); 1185479dd10fSwikidesign 1186479dd10fSwikidesign if (!@file_exists($cfile)) { 11872b18adb9SMichael Klier if ($this->getConf('automatic')) { 11882b18adb9SMichael Klier return true; 11892b18adb9SMichael Klier } else { 11902b18adb9SMichael Klier return false; 11912b18adb9SMichael Klier } 1192479dd10fSwikidesign } 1193479dd10fSwikidesign 1194479dd10fSwikidesign $comments = unserialize(io_readFile($cfile, false)); 1195479dd10fSwikidesign 119607c376bbSwikidesign if ($comments['title']) $title = hsc($comments['title']); 1197479dd10fSwikidesign $num = $comments['number']; 1198479dd10fSwikidesign if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return false; 1199479dd10fSwikidesign else return true; 1200479dd10fSwikidesign } 1201479dd10fSwikidesign 1202479dd10fSwikidesign /** 1203e7c760b3Swikidesign * Creates a new thread page 1204e7c760b3Swikidesign */ 12056046f25cSwikidesign function _newThread() { 120661437513Swikidesign global $ID, $INFO; 1207f0fda08aSwikidesign 12081ea794e5Swikidesign $ns = cleanID($_REQUEST['ns']); 1209f0fda08aSwikidesign $title = str_replace(':', '', $_REQUEST['title']); 12102e80cd5fSwikidesign $back = $ID; 12112e80cd5fSwikidesign $ID = ($ns ? $ns.':' : '').cleanID($title); 12122e80cd5fSwikidesign $INFO = pageinfo(); 1213f0fda08aSwikidesign 1214f0fda08aSwikidesign // check if we are allowed to create this file 12152e80cd5fSwikidesign if ($INFO['perm'] >= AUTH_CREATE) { 1216f0fda08aSwikidesign 1217f0fda08aSwikidesign //check if locked by anyone - if not lock for my self 12182e80cd5fSwikidesign if ($INFO['locked']) return 'locked'; 12192e80cd5fSwikidesign else lock($ID); 1220f0fda08aSwikidesign 1221f0fda08aSwikidesign // prepare the new thread file with default stuff 12222e80cd5fSwikidesign if (!@file_exists($INFO['filepath'])) { 1223f0fda08aSwikidesign global $TEXT; 1224f0fda08aSwikidesign 12252e80cd5fSwikidesign $TEXT = pageTemplate(array(($ns ? $ns.':' : '').$title)); 12261433886fSwikidesign if (!$TEXT) { 122761437513Swikidesign $data = array('id' => $ID, 'ns' => $ns, 'title' => $title, 'back' => $back); 122861437513Swikidesign $TEXT = $this->_pageTemplate($data); 12292e80cd5fSwikidesign } 12302e80cd5fSwikidesign return 'preview'; 1231f0fda08aSwikidesign } else { 12322e80cd5fSwikidesign return 'edit'; 1233f0fda08aSwikidesign } 1234f0fda08aSwikidesign } else { 12352e80cd5fSwikidesign return 'show'; 1236f0fda08aSwikidesign } 1237f0fda08aSwikidesign } 1238f0fda08aSwikidesign 1239e7c760b3Swikidesign /** 124061437513Swikidesign * Adapted version of pageTemplate() function 124161437513Swikidesign */ 124261437513Swikidesign function _pageTemplate($data) { 124361437513Swikidesign global $conf, $INFO; 124461437513Swikidesign 124561437513Swikidesign $id = $data['id']; 124661437513Swikidesign $user = $_SERVER['REMOTE_USER']; 124761437513Swikidesign $tpl = io_readFile(DOKU_PLUGIN.'discussion/_template.txt'); 124861437513Swikidesign 124961437513Swikidesign // standard replacements 125061437513Swikidesign $replace = array( 125161437513Swikidesign '@NS@' => $data['ns'], 125261437513Swikidesign '@PAGE@' => strtr(noNS($id),'_',' '), 125361437513Swikidesign '@USER@' => $user, 125461437513Swikidesign '@NAME@' => $INFO['userinfo']['name'], 125561437513Swikidesign '@MAIL@' => $INFO['userinfo']['mail'], 1256f014bc86SMichael Klier '@DATE@' => strftime($conf['dformat']), 125761437513Swikidesign ); 125861437513Swikidesign 125961437513Swikidesign // additional replacements 126061437513Swikidesign $replace['@BACK@'] = $data['back']; 126161437513Swikidesign $replace['@TITLE@'] = $data['title']; 126261437513Swikidesign 126361437513Swikidesign // avatar if useavatar and avatar plugin available 126461437513Swikidesign if ($this->getConf('useavatar') 126561437513Swikidesign && (@file_exists(DOKU_PLUGIN.'avatar/syntax.php')) 126661437513Swikidesign && (!plugin_isdisabled('avatar'))) { 126761437513Swikidesign $replace['@AVATAR@'] = '{{avatar>'.$user.' }} '; 126861437513Swikidesign } else { 126961437513Swikidesign $replace['@AVATAR@'] = ''; 127061437513Swikidesign } 127161437513Swikidesign 127261437513Swikidesign // tag if tag plugin is available 127361437513Swikidesign if ((@file_exists(DOKU_PLUGIN.'tag/syntax/tag.php')) 127461437513Swikidesign && (!plugin_isdisabled('tag'))) { 127561437513Swikidesign $replace['@TAG@'] = "\n\n{{tag>}}"; 127661437513Swikidesign } else { 127761437513Swikidesign $replace['@TAG@'] = ''; 127861437513Swikidesign } 127961437513Swikidesign 128061437513Swikidesign // do the replace 128161437513Swikidesign $tpl = str_replace(array_keys($replace), array_values($replace), $tpl); 128261437513Swikidesign return $tpl; 128361437513Swikidesign } 128461437513Swikidesign 128561437513Swikidesign /** 1286e7c760b3Swikidesign * Checks if the CAPTCHA string submitted is valid 1287e7c760b3Swikidesign * 1288e7c760b3Swikidesign * @author Andreas Gohr <gohr@cosmocode.de> 1289e7c760b3Swikidesign * @adaption Esther Brunner <wikidesign@gmail.com> 1290e7c760b3Swikidesign */ 12916046f25cSwikidesign function _captchaCheck() { 1292be1d1f2fSGina Haeussge if (plugin_isdisabled('captcha') || (!$captcha = plugin_load('helper', 'captcha'))) 1293be1d1f2fSGina Haeussge return; // CAPTCHA is disabled or not available 1294e7c760b3Swikidesign 1295d1c29589Swikidesign // do nothing if logged in user and no CAPTCHA required 1296d1c29589Swikidesign if (!$captcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return; 1297d1c29589Swikidesign 1298e7c760b3Swikidesign // compare provided string with decrypted captcha 1299e7c760b3Swikidesign $rand = PMA_blowfish_decrypt($_REQUEST['plugin__captcha_secret'], auth_cookiesalt()); 1300e7c760b3Swikidesign $code = $captcha->_generateCAPTCHA($captcha->_fixedIdent(), $rand); 1301e7c760b3Swikidesign 1302e7c760b3Swikidesign if (!$_REQUEST['plugin__captcha_secret'] || 1303e7c760b3Swikidesign !$_REQUEST['plugin__captcha'] || 1304e7c760b3Swikidesign strtoupper($_REQUEST['plugin__captcha']) != $code) { 1305e7c760b3Swikidesign 1306e7c760b3Swikidesign // CAPTCHA test failed! Continue to edit instead of saving 1307e7c760b3Swikidesign msg($captcha->getLang('testfailed'), -1); 1308e7c760b3Swikidesign if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit'; 1309e7c760b3Swikidesign elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show'; 1310e7c760b3Swikidesign } 1311e7c760b3Swikidesign // if we arrive here it was a valid save 1312e7c760b3Swikidesign } 1313e7c760b3Swikidesign 1314a1ca9e44Swikidesign /** 1315a1ca9e44Swikidesign * Adds the comments to the index 1316a1ca9e44Swikidesign */ 131761437513Swikidesign function idx_add_discussion(&$event, $param) { 1318a1ca9e44Swikidesign 1319a1ca9e44Swikidesign // get .comments meta file name 1320a1ca9e44Swikidesign $file = metaFN($event->data[0], '.comments'); 1321a1ca9e44Swikidesign 1322a1ca9e44Swikidesign if (@file_exists($file)) $data = unserialize(io_readFile($file, false)); 1323a1ca9e44Swikidesign if ((!$data['status']) || ($data['number'] == 0)) return; // comments are turned off 1324a1ca9e44Swikidesign 1325a1ca9e44Swikidesign // now add the comments 1326a1ca9e44Swikidesign if (isset($data['comments'])) { 1327a1ca9e44Swikidesign foreach ($data['comments'] as $key => $value) { 132877a22ba2Swikidesign $event->data[1] .= $this->_addCommentWords($key, $data); 1329a1ca9e44Swikidesign } 1330a1ca9e44Swikidesign } 1331a1ca9e44Swikidesign } 1332a1ca9e44Swikidesign 1333a1ca9e44Swikidesign /** 1334a1ca9e44Swikidesign * Adds the words of a given comment to the index 1335a1ca9e44Swikidesign */ 1336efbe59d0Swikidesign function _addCommentWords($cid, &$data, $parent = '') { 1337a1ca9e44Swikidesign 1338efbe59d0Swikidesign if (!isset($data['comments'][$cid])) return ''; // comment was removed 1339a1ca9e44Swikidesign $comment = $data['comments'][$cid]; 1340a1ca9e44Swikidesign 1341efbe59d0Swikidesign if (!is_array($comment)) return ''; // corrupt datatype 1342efbe59d0Swikidesign if ($comment['parent'] != $parent) return ''; // reply to an other comment 1343efbe59d0Swikidesign if (!$comment['show']) return ''; // hidden comment 1344a1ca9e44Swikidesign 1345efbe59d0Swikidesign $text = $comment['raw']; // we only add the raw comment text 1346efbe59d0Swikidesign if (is_array($comment['replies'])) { // and the replies 1347efbe59d0Swikidesign foreach ($comment['replies'] as $rid) { 1348efbe59d0Swikidesign $text .= $this->_addCommentWords($rid, $data, $cid); 1349a1ca9e44Swikidesign } 1350a1ca9e44Swikidesign } 1351efbe59d0Swikidesign return ' '.$text; 1352efbe59d0Swikidesign } 1353a10b5c98SMichael Klier 1354a10b5c98SMichael Klier /** 1355a10b5c98SMichael Klier * Only allow http(s) URLs and append http:// to URLs if needed 1356a10b5c98SMichael Klier */ 1357a10b5c98SMichael Klier function _checkURL($url) { 1358a10b5c98SMichael Klier if(preg_match("#^http://|^https://#", $url)) { 1359a10b5c98SMichael Klier return hsc($url); 1360a10b5c98SMichael Klier } elseif(substr($url, 0, 4) == 'www.') { 1361a10b5c98SMichael Klier return hsc('http://' . $url); 1362a10b5c98SMichael Klier } else { 1363a10b5c98SMichael Klier return ''; 1364a10b5c98SMichael Klier } 1365a10b5c98SMichael Klier } 1366f0fda08aSwikidesign} 136731aab30eSGina Haeussge 136831aab30eSGina Haeussgefunction _sortCallback($a, $b) { 136931aab30eSGina Haeussge if (is_array($a['date'])) { // new format 137031aab30eSGina Haeussge $createdA = $a['date']['created']; 137131aab30eSGina Haeussge } else { // old format 137231aab30eSGina Haeussge $createdA = $a['date']; 137331aab30eSGina Haeussge } 137431aab30eSGina Haeussge 137531aab30eSGina Haeussge if (is_array($b['date'])) { // new format 137631aab30eSGina Haeussge $createdB = $b['date']['created']; 137731aab30eSGina Haeussge } else { // old format 137831aab30eSGina Haeussge $createdB = $b['date']; 137931aab30eSGina Haeussge } 138031aab30eSGina Haeussge 138131aab30eSGina Haeussge if ($createdA == $createdB) 138231aab30eSGina Haeussge return 0; 138331aab30eSGina Haeussge else 138431aab30eSGina Haeussge return ($createdA < $createdB) ? -1 : 1; 138531aab30eSGina Haeussge} 138631aab30eSGina Haeussge 1387530693fbSMichael Klier// vim:ts=4:sw=4:et:enc=utf-8: 1388