1f0fda08aSwikidesign<?php 2f0fda08aSwikidesign/** 3f0fda08aSwikidesign * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 4f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 5f0fda08aSwikidesign */ 6f0fda08aSwikidesign 7f0fda08aSwikidesign// must be run within Dokuwiki 8f0fda08aSwikidesignif (!defined('DOKU_INC')) die(); 9f0fda08aSwikidesign 10573e23a1Swikidesignif (!defined('DOKU_LF')) define('DOKU_LF', "\n"); 11573e23a1Swikidesignif (!defined('DOKU_TAB')) define('DOKU_TAB', "\t"); 12f0fda08aSwikidesignif (!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/'); 13f0fda08aSwikidesign 14573e23a1Swikidesignrequire_once(DOKU_PLUGIN.'action.php'); 15f0fda08aSwikidesign 16f0fda08aSwikidesignclass action_plugin_discussion extends DokuWiki_Action_Plugin{ 17f0fda08aSwikidesign 18f1c6610eSpierre.spring var $avatar = null; 19f1c6610eSpierre.spring var $style = null; 20f1c6610eSpierre.spring var $use_avatar = null; 21f1c6610eSpierre.spring 22f0fda08aSwikidesign function getInfo() { 23f0fda08aSwikidesign return array( 244c4d91c6SGina Haeussge 'author' => 'Gina Häußge, Michael Klier, Esther Brunner', 254c4d91c6SGina Haeussge 'email' => 'dokuwiki@chimeric.de', 263648fb3eSMichael Klier 'date' => @file_get_contents(DOKU_PLUGIN.'discussion/VERSION'), 27573e23a1Swikidesign 'name' => 'Discussion Plugin (action component)', 28f0fda08aSwikidesign 'desc' => 'Enables discussion features', 294c4d91c6SGina Haeussge 'url' => 'http://wiki.splitbrain.org/plugin:discussion', 30f0fda08aSwikidesign ); 31f0fda08aSwikidesign } 32f0fda08aSwikidesign 33f0fda08aSwikidesign function register(&$contr) { 34f0fda08aSwikidesign $contr->register_hook( 35f0fda08aSwikidesign 'ACTION_ACT_PREPROCESS', 36f0fda08aSwikidesign 'BEFORE', 37f0fda08aSwikidesign $this, 38f0fda08aSwikidesign 'handle_act_preprocess', 39f0fda08aSwikidesign array() 40f0fda08aSwikidesign ); 41f0fda08aSwikidesign $contr->register_hook( 42f0fda08aSwikidesign 'TPL_ACT_RENDER', 43f0fda08aSwikidesign 'AFTER', 44f0fda08aSwikidesign $this, 45f0fda08aSwikidesign 'comments', 46f0fda08aSwikidesign array() 47f0fda08aSwikidesign ); 48479dd10fSwikidesign $contr->register_hook( 49a1ca9e44Swikidesign 'INDEXER_PAGE_ADD', 50a1ca9e44Swikidesign 'AFTER', 51a1ca9e44Swikidesign $this, 5261437513Swikidesign 'idx_add_discussion', 53a1ca9e44Swikidesign array() 54a1ca9e44Swikidesign ); 552d4bee9aSMichael Klier $contr->register_hook( 562d4bee9aSMichael Klier 'TPL_METAHEADER_OUTPUT', 572d4bee9aSMichael Klier 'BEFORE', 582d4bee9aSMichael Klier $this, 592d4bee9aSMichael Klier 'handle_tpl_metaheader_output', 602d4bee9aSMichael Klier array() 612d4bee9aSMichael Klier ); 622d4bee9aSMichael Klier $contr->register_hook( 632d4bee9aSMichael Klier 'TOOLBAR_DEFINE', 642d4bee9aSMichael Klier 'AFTER', 652d4bee9aSMichael Klier $this, 662d4bee9aSMichael Klier 'handle_toolbar_define', 672d4bee9aSMichael Klier array() 682d4bee9aSMichael Klier ); 692d4bee9aSMichael Klier } 702d4bee9aSMichael Klier 712d4bee9aSMichael Klier /** 722d4bee9aSMichael Klier * Modify Tollbar for use with discussion plugin 732d4bee9aSMichael Klier * 742d4bee9aSMichael Klier * @author Michael Klier <chi@chimeric.de> 752d4bee9aSMichael Klier */ 762d4bee9aSMichael Klier function handle_toolbar_define(&$event, $param) { 772d4bee9aSMichael Klier global $ACT; 782d4bee9aSMichael Klier if($ACT != 'show') return; 792d4bee9aSMichael Klier 801de52da1SMichael Klier if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) { 812d4bee9aSMichael Klier $toolbar = array(); 822d4bee9aSMichael Klier foreach($event->data as $btn) { 832d4bee9aSMichael Klier if($btn['type'] == 'mediapopup') continue; 842d4bee9aSMichael Klier if($btn['type'] == 'signature') continue; 852d4bee9aSMichael Klier if(preg_match("/=+?/", $btn['open'])) continue; 862d4bee9aSMichael Klier array_push($toolbar, $btn); 872d4bee9aSMichael Klier } 882d4bee9aSMichael Klier $event->data = $toolbar; 892d4bee9aSMichael Klier } 902d4bee9aSMichael Klier } 912d4bee9aSMichael Klier 922d4bee9aSMichael Klier /** 932d4bee9aSMichael Klier * Dirty workaround to add a toolbar to the discussion plugin 942d4bee9aSMichael Klier * 952d4bee9aSMichael Klier * @author Michael Klier <chi@chimeric.de> 962d4bee9aSMichael Klier */ 972d4bee9aSMichael Klier function handle_tpl_metaheader_output(&$event, $param) { 982d4bee9aSMichael Klier global $ACT; 992d4bee9aSMichael Klier global $ID; 1002d4bee9aSMichael Klier if($ACT != 'show') return; 1012d4bee9aSMichael Klier 1022d4bee9aSMichael Klier // FIXME check if this works for global discussion/on too 1031de52da1SMichael Klier if($this->_hasDiscussion($title) && $this->getConf('wikisyntaxok')) { 1042d4bee9aSMichael Klier // FIXME ugly workaround, replace this once DW the toolbar code is more flexible 1052d4bee9aSMichael Klier array_unshift($event->data['script'], array('type' => 'text/javascript', 'charset' => 'utf-8', '_data' => '', 'src' => DOKU_BASE.'lib/scripts/edit.js')); 1062d4bee9aSMichael Klier @require_once(DOKU_INC.'inc/toolbar.php'); 1072d4bee9aSMichael Klier ob_start(); 1082d4bee9aSMichael Klier print 'NS = "' . getNS($ID) . '";'; // we have to define NS, otherwise we get get JS errors 1092d4bee9aSMichael Klier toolbar_JSdefines('toolbar'); 1102d4bee9aSMichael Klier $script = ob_get_clean(); 1112d4bee9aSMichael Klier array_push($event->data['script'], array('type' => 'text/javascript', 'charset' => "utf-8", '_data' => $script)); 1122d4bee9aSMichael Klier } 113f0fda08aSwikidesign } 114f0fda08aSwikidesign 115f0fda08aSwikidesign /** 116a1d93126SGina Haeussge * Handles comment actions, dispatches data processing routines 117f0fda08aSwikidesign */ 118a1d93126SGina Haeussge function handle_act_preprocess(&$event, $param) { 119284f9aa2SMichael Klier global $ID; 120d7eb2e24SMichael Klier global $INFO; 121284f9aa2SMichael Klier global $conf; 1223011fb8bSMichael Klier global $lang; 123573e23a1Swikidesign 124a1d93126SGina Haeussge // handle newthread ACTs 125a1d93126SGina Haeussge if ($event->data == 'newthread') { 126a1d93126SGina Haeussge // we can handle it -> prevent others 127a1d93126SGina Haeussge $event->preventDefault(); 128a1d93126SGina Haeussge $event->data = $this->_newThread(); 129a1d93126SGina Haeussge } 130a1d93126SGina Haeussge 131a1d93126SGina Haeussge // enable captchas 132a1d93126SGina Haeussge if ((in_array($_REQUEST['comment'], array('add', 'save'))) 133a1d93126SGina Haeussge && (@file_exists(DOKU_PLUGIN.'captcha/action.php'))) { 134a1d93126SGina Haeussge $this->_captchaCheck(); 135a1d93126SGina Haeussge } 136a1d93126SGina Haeussge 1373011fb8bSMichael Klier // if we are not in show mode or someone wants to unsubscribe, that was all for now 1389881d835SMichael Klier if ($event->data != 'show' && $event->data != 'unsubscribe' && $event->data != 'confirmsubscribe') return; 139a1d93126SGina Haeussge 1409881d835SMichael Klier if ($event->data == 'unsubscribe' or $event->data == 'confirmsubscribe') { 1419881d835SMichael Klier // ok we can handle it prevent others 1429881d835SMichael Klier $event->preventDefault(); 1439881d835SMichael Klier 1443011fb8bSMichael Klier if (!isset($_REQUEST['hash'])) { 1459881d835SMichael Klier return false; 1463011fb8bSMichael Klier } else { 1473011fb8bSMichael Klier $file = metaFN($ID, '.comments'); 1483011fb8bSMichael Klier $data = unserialize(io_readFile($file)); 1499881d835SMichael Klier foreach($data['subscribers'] as $mail => $info) { 1509881d835SMichael Klier // convert old style subscribers just in case 1519881d835SMichael Klier if(!is_array($info)) { 1529881d835SMichael Klier $hash = $data['subscribers'][$mail]; 1539881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 1549881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 1559881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 1569881d835SMichael Klier } 1579881d835SMichael Klier } 1589881d835SMichael Klier 1599881d835SMichael Klier if($data['subscribers'][$mail]['hash'] == $_REQUEST['hash']) { 1609881d835SMichael Klier if($event->data == 'unsubscribe') { 1613011fb8bSMichael Klier unset($data['subscribers'][$mail]); 1623011fb8bSMichael Klier msg(sprintf($lang['unsubscribe_success'], $mail, $ID), 1); 1639881d835SMichael Klier } elseif($event->data == 'confirmsubscribe') { 1649881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 1659881d835SMichael Klier msg(sprintf($lang['subscribe_success'], $mail, $ID), 1); 1669881d835SMichael Klier } 1679881d835SMichael Klier io_saveFile($file, serialize($data)); 1683011fb8bSMichael Klier $event->data = 'show'; 1693011fb8bSMichael Klier return true; 1709881d835SMichael Klier } else { 1713011fb8bSMichael Klier return false; 1723011fb8bSMichael Klier } 1739881d835SMichael Klier } 1743011fb8bSMichael Klier } else { 175a1d93126SGina Haeussge // do the data processing for comments 176f0fda08aSwikidesign $cid = $_REQUEST['cid']; 177f0fda08aSwikidesign switch ($_REQUEST['comment']) { 178f0fda08aSwikidesign case 'add': 17937e3c825SMichael Klier if(empty($_REQUEST['text'])) return; // don't add empty comments 18094c5d164SMichael Klier if(isset($_SERVER['REMOTE_USER']) && !$this->getConf('adminimport')) { 18194c5d164SMichael Klier $comment['user']['id'] = $_SERVER['REMOTE_USER']; 18294c5d164SMichael Klier $comment['user']['name'] = $INFO['userinfo']['name']; 18394c5d164SMichael Klier $comment['user']['mail'] = $INFO['userinfo']['mail']; 18494c5d164SMichael Klier } elseif((isset($_SERVER['REMOTE_USER']) && $this->getConf('adminimport') && auth_ismanager()) || !isset($_SERVER['REMOTE_USER'])) { 18537e3c825SMichael Klier if(empty($_REQUEST['name']) or empty($_REQUEST['mail'])) return // don't add anonymous comments 18694c5d164SMichael Klier $comment['user']['id'] = 'test'.hsc($_REQUEST['user']); 18794c5d164SMichael Klier $comment['user']['name'] = hsc($_REQUEST['name']); 18894c5d164SMichael Klier $comment['user']['mail'] = hsc($_REQUEST['mail']); 18994c5d164SMichael Klier } 19094c5d164SMichael Klier $comment['user']['address'] = ($this->getConf('addressfield')) ? hsc($_REQUEST['address']) : ''; 191a10b5c98SMichael Klier $comment['user']['url'] = ($this->getConf('urlfield')) ? $this->_checkURL($_REQUEST['url']) : ''; 19294c5d164SMichael Klier $comment['subscribe'] = ($this->getConf('subscribe')) ? $_REQUEST['subscribe'] : ''; 19394c5d164SMichael Klier $comment['date'] = array('created' => $_REQUEST['date']); 19494c5d164SMichael Klier $comment['raw'] = cleanText($_REQUEST['text']); 195f0fda08aSwikidesign $repl = $_REQUEST['reply']; 196b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 197a44bc9f7SMichael Klier $comment['show'] = false; 198a44bc9f7SMichael Klier } else { 199a44bc9f7SMichael Klier $comment['show'] = true; 200a44bc9f7SMichael Klier } 201f0fda08aSwikidesign $this->_add($comment, $repl); 202f0fda08aSwikidesign break; 203f0fda08aSwikidesign 204f0fda08aSwikidesign case 'save': 205f0fda08aSwikidesign $raw = cleanText($_REQUEST['text']); 206264b7327Swikidesign $this->_save(array($cid), $raw); 207f0fda08aSwikidesign break; 208f0fda08aSwikidesign 2091e46d176Swikidesign case 'delete': 210264b7327Swikidesign $this->_save(array($cid), ''); 2112ee3dca3Swikidesign break; 2121e46d176Swikidesign 213f0fda08aSwikidesign case 'toogle': 214264b7327Swikidesign $this->_save(array($cid), '', 'toogle'); 215f0fda08aSwikidesign break; 216a1d93126SGina Haeussge } 2173011fb8bSMichael Klier } 218284f9aa2SMichael Klier 219284f9aa2SMichael Klier // FIXME use new TPL_TOC_RENDER event in the future 220d7eb2e24SMichael Klier if(count($INFO['meta']['description']['tableofcontents']) >= ($conf['maxtoclevel']-1) && $INFO['meta']['internal']['toc']) { 221284f9aa2SMichael Klier 222284f9aa2SMichael Klier $TOC = array(); 223284f9aa2SMichael Klier global $TOC; 224d7eb2e24SMichael Klier $TOC = $INFO['meta']['description']['tableofcontents']; 225284f9aa2SMichael Klier 226284f9aa2SMichael Klier $tocitem = array( 'hid' => 'discussion__section', 227284f9aa2SMichael Klier 'title' => $this->getLang('discussion'), 228284f9aa2SMichael Klier 'type' => 'ul', 229284f9aa2SMichael Klier 'level' => 1 ); 230284f9aa2SMichael Klier 231284f9aa2SMichael Klier $file = metaFN($ID, '.comments'); 232284f9aa2SMichael Klier if(@file_exists($file)) { 233284f9aa2SMichael Klier $data = unserialize(io_readFile($file)); 234284f9aa2SMichael Klier if($data['status'] != 0 && !empty($TOC)) { 235284f9aa2SMichael Klier $TOC[] = $tocitem; 236284f9aa2SMichael Klier } 237284f9aa2SMichael Klier } 238284f9aa2SMichael Klier } 239a1d93126SGina Haeussge } 240a1d93126SGina Haeussge 241a1d93126SGina Haeussge /** 242a1d93126SGina Haeussge * Main function; dispatches the visual comment actions 243a1d93126SGina Haeussge */ 244a1d93126SGina Haeussge function comments(&$event, $param) { 245a1d93126SGina Haeussge if ($event->data != 'show') return; // nothing to do for us 246a1d93126SGina Haeussge 247a1d93126SGina Haeussge $cid = $_REQUEST['cid']; 248a1d93126SGina Haeussge switch ($_REQUEST['comment']) { 249a1d93126SGina Haeussge case 'edit': 250a1d93126SGina Haeussge $this->_show(NULL, $cid); 251a1d93126SGina Haeussge break; 2522b18adb9SMichael Klier default: 253f0fda08aSwikidesign $this->_show($cid); 2542b18adb9SMichael Klier break; 255f0fda08aSwikidesign } 256f0fda08aSwikidesign } 257f0fda08aSwikidesign 258f0fda08aSwikidesign /** 259a1d93126SGina Haeussge * Redirects browser to given comment anchor 260a1d93126SGina Haeussge */ 261a1d93126SGina Haeussge function _redirect($cid) { 262a1d93126SGina Haeussge global $ID; 263a1d93126SGina Haeussge global $ACT; 264a1d93126SGina Haeussge 265a1d93126SGina Haeussge if ($ACT !== 'show') return; 266a44bc9f7SMichael Klier 267b42e13e2SMichael Klier if($this->getConf('moderate') && !auth_ismanager()) { 268a44bc9f7SMichael Klier msg($this->getLang('moderation'), 1); 269a44bc9f7SMichael Klier @session_start(); 270a44bc9f7SMichael Klier global $MSG; 271a44bc9f7SMichael Klier $_SESSION[DOKU_COOKIE]['msg'] = $MSG; 272a44bc9f7SMichael Klier session_write_close(); 273a44bc9f7SMichael Klier $url = wl($ID); 274a44bc9f7SMichael Klier } else { 275a44bc9f7SMichael Klier $url = wl($ID) . '#comment_' . $cid; 276a44bc9f7SMichael Klier } 277a44bc9f7SMichael Klier send_redirect($url); 2786d1f4f20SGina Haeussge exit(); 279a1d93126SGina Haeussge } 280a1d93126SGina Haeussge 281a1d93126SGina Haeussge /** 282f0fda08aSwikidesign * Shows all comments of the current page 283f0fda08aSwikidesign */ 284f0fda08aSwikidesign function _show($reply = NULL, $edit = NULL) { 2852b18adb9SMichael Klier global $ID; 2862b18adb9SMichael Klier global $INFO; 2872b18adb9SMichael Klier global $ACT; 288573e23a1Swikidesign 289479dd10fSwikidesign // get .comments meta file name 290f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 291f0fda08aSwikidesign 2924aeddd8eSMichael Klier if (!$INFO['exists']) return; 2932b18adb9SMichael Klier if (!@file_exists($file) && !$this->getConf('automatic')) return false; 294f0fda08aSwikidesign 2952b18adb9SMichael Klier // load data 2962b18adb9SMichael Klier if (@file_exists($file)) { 2972b18adb9SMichael Klier $data = unserialize(io_readFile($file, false)); 298479dd10fSwikidesign if (!$data['status']) return false; // comments are turned off 2992b18adb9SMichael Klier } elseif (!@file_exists($file) && $this->getConf('automatic') && $INFO['exists']) { 3002b18adb9SMichael Klier // set status to show the comment form 3012b18adb9SMichael Klier $data['status'] = 1; 3022b18adb9SMichael Klier $data['number'] = 0; 3032b18adb9SMichael Klier } 304f0fda08aSwikidesign 305f0fda08aSwikidesign // section title 30607c376bbSwikidesign $title = ($data['title'] ? hsc($data['title']) : $this->getLang('discussion')); 3074a0a1bd2Swikidesign ptln('<div class="comment_wrapper">'); 3084a0a1bd2Swikidesign ptln('<h2><a name="discussion__section" id="discussion__section">', 2); 3094a0a1bd2Swikidesign ptln($title, 4); 3104a0a1bd2Swikidesign ptln('</a></h2>', 2); 3114a0a1bd2Swikidesign ptln('<div class="level2 hfeed">', 2); 312f0fda08aSwikidesign // now display the comments 313f0fda08aSwikidesign if (isset($data['comments'])) { 31431aab30eSGina Haeussge if (!$this->getConf('usethreading')) { 31531aab30eSGina Haeussge $data['comments'] = $this->_flattenThreads($data['comments']); 31631aab30eSGina Haeussge uasort($data['comments'], '_sortCallBack'); 31731aab30eSGina Haeussge } 318dbd9d5cdSMichael Klier if($this->getConf('newestfirst')) { 319dbd9d5cdSMichael Klier $data['comments'] = array_reverse($data['comments']); 320dbd9d5cdSMichael Klier } 321f0fda08aSwikidesign foreach ($data['comments'] as $key => $value) { 322f0fda08aSwikidesign if ($key == $edit) $this->_form($value['raw'], 'save', $edit); // edit form 323f0fda08aSwikidesign else $this->_print($key, $data, '', $reply); 324f0fda08aSwikidesign } 325f0fda08aSwikidesign } 326f0fda08aSwikidesign 327f0fda08aSwikidesign // comment form 32831aab30eSGina Haeussge if (($data['status'] == 1) && (!$reply || !$this->getConf('usethreading')) && !$edit) $this->_form(''); 329f0fda08aSwikidesign 3304a0a1bd2Swikidesign ptln('</div>', 2); // level2 hfeed 3314a0a1bd2Swikidesign ptln('</div>'); // comment_wrapper 332f0fda08aSwikidesign 333f0fda08aSwikidesign return true; 334f0fda08aSwikidesign } 335f0fda08aSwikidesign 33631aab30eSGina Haeussge function _flattenThreads($comments, $keys = null) { 33731aab30eSGina Haeussge if (is_null($keys)) 33831aab30eSGina Haeussge $keys = array_keys($comments); 33931aab30eSGina Haeussge 34031aab30eSGina Haeussge foreach($keys as $cid) { 34131aab30eSGina Haeussge if (!empty($comments[$cid]['replies'])) { 34231aab30eSGina Haeussge $rids = $comments[$cid]['replies']; 34331aab30eSGina Haeussge $comments = $this->_flattenThreads($comments, $rids); 34431aab30eSGina Haeussge $comments[$cid]['replies'] = array(); 34531aab30eSGina Haeussge } 34631aab30eSGina Haeussge $comments[$cid]['parent'] = ''; 34731aab30eSGina Haeussge } 34831aab30eSGina Haeussge return $comments; 34931aab30eSGina Haeussge } 35031aab30eSGina Haeussge 351f0fda08aSwikidesign /** 352f0fda08aSwikidesign * Adds a new comment and then displays all comments 353f0fda08aSwikidesign */ 354f0fda08aSwikidesign function _add($comment, $parent) { 3553011fb8bSMichael Klier global $lang; 3563011fb8bSMichael Klier global $ID; 3573011fb8bSMichael Klier global $TEXT; 358f0fda08aSwikidesign 359f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 360f0fda08aSwikidesign $TEXT = $comment['raw']; 361f0fda08aSwikidesign 362f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 363f0fda08aSwikidesign if (checkwordblock()) { 364f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 365f0fda08aSwikidesign return false; 366f0fda08aSwikidesign } 367f0fda08aSwikidesign 3683011fb8bSMichael Klier if ((!$this->getConf('allowguests')) 3693011fb8bSMichael Klier && ($comment['user']['id'] != $_SERVER['REMOTE_USER'])) 3703011fb8bSMichael Klier return false; // guest comments not allowed 3713011fb8bSMichael Klier 372f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 373f0fda08aSwikidesign 374f0fda08aSwikidesign // get discussion meta file name 375f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 376f0fda08aSwikidesign 3772b18adb9SMichael Klier // create comments file if it doesn't exist yet 3782b18adb9SMichael Klier if(!@file_exists($file)) { 3792b18adb9SMichael Klier $data = array('status' => 1, 'number' => 0); 3802b18adb9SMichael Klier io_saveFile($file, serialize($data)); 3812b18adb9SMichael Klier } else { 382f0fda08aSwikidesign $data = array(); 383f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 384f0fda08aSwikidesign if ($data['status'] != 1) return false; // comments off or closed 3852b18adb9SMichael Klier } 3862b18adb9SMichael Klier 3873011fb8bSMichael Klier if ($comment['date']['created']) { 3883011fb8bSMichael Klier $date = strtotime($comment['date']['created']); 3893011fb8bSMichael Klier } else { 3903011fb8bSMichael Klier $date = time(); 3913011fb8bSMichael Klier } 392f0fda08aSwikidesign 3933011fb8bSMichael Klier if ($date == -1) { 3943011fb8bSMichael Klier $date = time(); 3953011fb8bSMichael Klier } 3963011fb8bSMichael Klier 3976046f25cSwikidesign $cid = md5($comment['user']['id'].$date); // create a unique id 398f0fda08aSwikidesign 3993011fb8bSMichael Klier if (!is_array($data['comments'][$parent])) { 4003011fb8bSMichael Klier $parent = NULL; // invalid parent comment 4013011fb8bSMichael Klier } 402f0fda08aSwikidesign 403f0fda08aSwikidesign // render the comment 404f0fda08aSwikidesign $xhtml = $this->_render($comment['raw']); 405f0fda08aSwikidesign 406f0fda08aSwikidesign // fill in the new comment 407f0fda08aSwikidesign $data['comments'][$cid] = array( 4086046f25cSwikidesign 'user' => $comment['user'], 4096046f25cSwikidesign 'date' => array('created' => $date), 410f0fda08aSwikidesign 'show' => true, 4116046f25cSwikidesign 'raw' => $comment['raw'], 412f0fda08aSwikidesign 'xhtml' => $xhtml, 413f0fda08aSwikidesign 'parent' => $parent, 414a44bc9f7SMichael Klier 'replies' => array(), 415a44bc9f7SMichael Klier 'show' => $comment['show'] 416f0fda08aSwikidesign ); 417f0fda08aSwikidesign 4183011fb8bSMichael Klier if($comment['subscribe']) { 4193011fb8bSMichael Klier $mail = $comment['user']['mail']; 4203011fb8bSMichael Klier if($data['subscribers']) { 4213011fb8bSMichael Klier if(!$data['subscribers'][$mail]) { 4229881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4239881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4249881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4259881d835SMichael Klier } else { 4269881d835SMichael Klier // convert old style subscribers and set them active 4279881d835SMichael Klier if(!is_array($data['subscribers'][$mail])) { 4289881d835SMichael Klier $hash = $data['subscribers'][$mail]; 4299881d835SMichael Klier $data['subscribers'][$mail]['hash'] = $hash; 4309881d835SMichael Klier $data['subscribers'][$mail]['active'] = true; 4319881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = true; 4329881d835SMichael Klier } 4333011fb8bSMichael Klier } 4343011fb8bSMichael Klier } else { 4359881d835SMichael Klier $data['subscribers'][$mail]['hash'] = md5($mail . mt_rand()); 4369881d835SMichael Klier $data['subscribers'][$mail]['active'] = false; 4379881d835SMichael Klier $data['subscribers'][$mail]['confirmsent'] = false; 4383011fb8bSMichael Klier } 4393011fb8bSMichael Klier } 4403011fb8bSMichael Klier 441f0fda08aSwikidesign // update parent comment 442f0fda08aSwikidesign if ($parent) $data['comments'][$parent]['replies'][] = $cid; 443f0fda08aSwikidesign 444f0fda08aSwikidesign // update the number of comments 445f0fda08aSwikidesign $data['number']++; 446f0fda08aSwikidesign 447f0fda08aSwikidesign // notify subscribers of the page 4488b42cefbSMichael Klier $data['comments'][$cid]['cid'] = $cid; 4493011fb8bSMichael Klier $this->_notify($data['comments'][$cid], $data['subscribers']); 450f0fda08aSwikidesign 4519881d835SMichael Klier // save the comment metadata file 4529881d835SMichael Klier io_saveFile($file, serialize($data)); 4539881d835SMichael Klier $this->_addLogEntry($date, $ID, 'cc', '', $cid); 4549881d835SMichael Klier 455a1d93126SGina Haeussge $this->_redirect($cid); 456f0fda08aSwikidesign return true; 457f0fda08aSwikidesign } 458f0fda08aSwikidesign 459f0fda08aSwikidesign /** 460f0fda08aSwikidesign * Saves the comment with the given ID and then displays all comments 461f0fda08aSwikidesign */ 462264b7327Swikidesign function _save($cids, $raw, $act = NULL) { 463a1ace3c9Swikidesign global $ID; 464f0fda08aSwikidesign 4652ee3dca3Swikidesign if ($raw) { 4662ee3dca3Swikidesign global $TEXT; 4672ee3dca3Swikidesign 468f0fda08aSwikidesign $otxt = $TEXT; // set $TEXT to comment text for wordblock check 469f0fda08aSwikidesign $TEXT = $raw; 470f0fda08aSwikidesign 471f0fda08aSwikidesign // spamcheck against the DokuWiki blacklist 472f0fda08aSwikidesign if (checkwordblock()) { 473f0fda08aSwikidesign msg($this->getLang('wordblock'), -1); 474f0fda08aSwikidesign return false; 475f0fda08aSwikidesign } 476f0fda08aSwikidesign 477f0fda08aSwikidesign $TEXT = $otxt; // restore global $TEXT 4782ee3dca3Swikidesign } 479f0fda08aSwikidesign 480f0fda08aSwikidesign // get discussion meta file name 481f0fda08aSwikidesign $file = metaFN($ID, '.comments'); 482f0fda08aSwikidesign $data = unserialize(io_readFile($file, false)); 483f0fda08aSwikidesign 484573e23a1Swikidesign if (!is_array($cids)) $cids = array($cids); 485264b7327Swikidesign foreach ($cids as $cid) { 486264b7327Swikidesign 4876046f25cSwikidesign if (is_array($data['comments'][$cid]['user'])) { 4886046f25cSwikidesign $user = $data['comments'][$cid]['user']['id']; 4896046f25cSwikidesign $convert = false; 4906046f25cSwikidesign } else { 4916046f25cSwikidesign $user = $data['comments'][$cid]['user']; 4926046f25cSwikidesign $convert = true; 4936046f25cSwikidesign } 4946046f25cSwikidesign 495f0fda08aSwikidesign // someone else was trying to edit our comment -> abort 4963e02b3ffSwikidesign if (($user != $_SERVER['REMOTE_USER']) && (!auth_ismanager())) return false; 497f0fda08aSwikidesign 498f0fda08aSwikidesign $date = time(); 499f0fda08aSwikidesign 5006046f25cSwikidesign // need to convert to new format? 5016046f25cSwikidesign if ($convert) { 5026046f25cSwikidesign $data['comments'][$cid]['user'] = array( 5036046f25cSwikidesign 'id' => $user, 5046046f25cSwikidesign 'name' => $data['comments'][$cid]['name'], 5056046f25cSwikidesign 'mail' => $data['comments'][$cid]['mail'], 5066046f25cSwikidesign 'url' => $data['comments'][$cid]['url'], 5076046f25cSwikidesign 'address' => $data['comments'][$cid]['address'], 5086046f25cSwikidesign ); 5096046f25cSwikidesign $data['comments'][$cid]['date'] = array( 5106046f25cSwikidesign 'created' => $data['comments'][$cid]['date'] 5116046f25cSwikidesign ); 5126046f25cSwikidesign } 5136046f25cSwikidesign 514264b7327Swikidesign if ($act == 'toogle') { // toogle visibility 515f0fda08aSwikidesign $now = $data['comments'][$cid]['show']; 516f0fda08aSwikidesign $data['comments'][$cid]['show'] = !$now; 517f0fda08aSwikidesign $data['number'] = $this->_count($data); 518f0fda08aSwikidesign 519f0fda08aSwikidesign $type = ($data['comments'][$cid]['show'] ? 'sc' : 'hc'); 520f0fda08aSwikidesign 521264b7327Swikidesign } elseif ($act == 'show') { // show comment 522264b7327Swikidesign $data['comments'][$cid]['show'] = true; 523264b7327Swikidesign $data['number'] = $this->_count($data); 524264b7327Swikidesign 525573e23a1Swikidesign $type = 'sc'; // show comment 526264b7327Swikidesign 527264b7327Swikidesign } elseif ($act == 'hide') { // hide comment 528264b7327Swikidesign $data['comments'][$cid]['show'] = false; 529264b7327Swikidesign $data['number'] = $this->_count($data); 530264b7327Swikidesign 531573e23a1Swikidesign $type = 'hc'; // hide comment 532264b7327Swikidesign 533f0fda08aSwikidesign } elseif (!$raw) { // remove the comment 534efbe59d0Swikidesign $data['comments'] = $this->_removeComment($cid, $data['comments']); 535f0fda08aSwikidesign $data['number'] = $this->_count($data); 536f0fda08aSwikidesign 537573e23a1Swikidesign $type = 'dc'; // delete comment 538f0fda08aSwikidesign 539f0fda08aSwikidesign } else { // save changed comment 540f0fda08aSwikidesign $xhtml = $this->_render($raw); 541f0fda08aSwikidesign 542f0fda08aSwikidesign // now change the comment's content 5436046f25cSwikidesign $data['comments'][$cid]['date']['modified'] = $date; 5446046f25cSwikidesign $data['comments'][$cid]['raw'] = $raw; 545f0fda08aSwikidesign $data['comments'][$cid]['xhtml'] = $xhtml; 546f0fda08aSwikidesign 547573e23a1Swikidesign $type = 'ec'; // edit comment 548f0fda08aSwikidesign } 549264b7327Swikidesign } 550264b7327Swikidesign 551f0fda08aSwikidesign // save the comment metadata file 552f0fda08aSwikidesign io_saveFile($file, serialize($data)); 553f0fda08aSwikidesign $this->_addLogEntry($date, $ID, $type, '', $cid); 554f0fda08aSwikidesign 555a1d93126SGina Haeussge $this->_redirect($cid); 556f0fda08aSwikidesign return true; 557f0fda08aSwikidesign } 558f0fda08aSwikidesign 559f0fda08aSwikidesign /** 560efbe59d0Swikidesign * Recursive function to remove a comment 561efbe59d0Swikidesign */ 562efbe59d0Swikidesign function _removeComment($cid, $comments) { 563efbe59d0Swikidesign if (is_array($comments[$cid]['replies'])) { 564efbe59d0Swikidesign foreach ($comments[$cid]['replies'] as $rid) { 565efbe59d0Swikidesign $comments = $this->_removeComment($rid, $comments); 566efbe59d0Swikidesign } 567efbe59d0Swikidesign } 568efbe59d0Swikidesign unset($comments[$cid]); 569efbe59d0Swikidesign return $comments; 570efbe59d0Swikidesign } 571efbe59d0Swikidesign 572efbe59d0Swikidesign /** 573f0fda08aSwikidesign * Prints an individual comment 574f0fda08aSwikidesign */ 575f0fda08aSwikidesign function _print($cid, &$data, $parent = '', $reply = '', $visible = true) { 576f0fda08aSwikidesign 5772ee3dca3Swikidesign if (!isset($data['comments'][$cid])) return false; // comment was removed 578f0fda08aSwikidesign $comment = $data['comments'][$cid]; 579f0fda08aSwikidesign 580f0fda08aSwikidesign if (!is_array($comment)) return false; // corrupt datatype 581f0fda08aSwikidesign 582f0fda08aSwikidesign if ($comment['parent'] != $parent) return true; // reply to an other comment 583f0fda08aSwikidesign 584f0fda08aSwikidesign if (!$comment['show']) { // comment hidden 585a1ace3c9Swikidesign if (auth_ismanager()) $hidden = ' comment_hidden'; 586f0fda08aSwikidesign else return true; 5874a0a1bd2Swikidesign } else { 5884a0a1bd2Swikidesign $hidden = ''; 589f0fda08aSwikidesign } 590f0fda08aSwikidesign 591f1c6610eSpierre.spring if($this->getConf('newestfirst')) { 592f1c6610eSpierre.spring // reply form 593f1c6610eSpierre.spring $this->_print_form($cid, $reply); 594f1c6610eSpierre.spring // replies to this comment entry? 595f1c6610eSpierre.spring $this->_print_replies($cid, $data, $reply, $visible); 596f1c6610eSpierre.spring // print the actual comment 597*6b1c1263SMichael Klier $this->_print_comment($cid, $data, $parent, $reply, $visible, $hidden); 598f1c6610eSpierre.spring } else { 599f1c6610eSpierre.spring // print the actual comment 600*6b1c1263SMichael Klier $this->_print_comment($cid, $data, $parent, $reply, $visible, $hidden); 601f1c6610eSpierre.spring // replies to this comment entry? 602f1c6610eSpierre.spring $this->_print_replies($cid, $data, $reply, $visible); 603f1c6610eSpierre.spring // reply form 604f1c6610eSpierre.spring $this->_print_form($cid, $reply); 605f1c6610eSpierre.spring } 606f1c6610eSpierre.spring } 607f1c6610eSpierre.spring 608f1c6610eSpierre.spring function _print_comment($cid, &$data, $parent, $reply, $visible, $hidden) 609f1c6610eSpierre.spring { 610f1c6610eSpierre.spring global $conf, $lang, $ID, $HIGH; 611f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 612f1c6610eSpierre.spring 613f0fda08aSwikidesign // comment head with date and user data 6144a0a1bd2Swikidesign ptln('<div class="hentry'.$hidden.'">', 4); 6154a0a1bd2Swikidesign ptln('<div class="comment_head">', 6); 6161810ba9cSMichael Klier ptln('<a name="comment_'.$cid.'" id="comment_'.$cid.'"></a>', 8); 6174a0a1bd2Swikidesign $head = '<span class="vcard author">'; 618f0fda08aSwikidesign 6196046f25cSwikidesign // prepare variables 6206046f25cSwikidesign if (is_array($comment['user'])) { // new format 6216046f25cSwikidesign $user = $comment['user']['id']; 6226046f25cSwikidesign $name = $comment['user']['name']; 6236046f25cSwikidesign $mail = $comment['user']['mail']; 6246046f25cSwikidesign $url = $comment['user']['url']; 6256046f25cSwikidesign $address = $comment['user']['address']; 6266046f25cSwikidesign } else { // old format 6276046f25cSwikidesign $user = $comment['user']; 6286046f25cSwikidesign $name = $comment['name']; 6296046f25cSwikidesign $mail = $comment['mail']; 6306046f25cSwikidesign $url = $comment['url']; 6316046f25cSwikidesign $address = $comment['address']; 6326046f25cSwikidesign } 6336046f25cSwikidesign if (is_array($comment['date'])) { // new format 6346046f25cSwikidesign $created = $comment['date']['created']; 6356046f25cSwikidesign $modified = $comment['date']['modified']; 6366046f25cSwikidesign } else { // old format 6376046f25cSwikidesign $created = $comment['date']; 6386046f25cSwikidesign $modified = $comment['edited']; 6396046f25cSwikidesign } 6406046f25cSwikidesign 641ce7b17cfSwikidesign // show avatar image? 642f1c6610eSpierre.spring if ($this->_use_avatar()) { 64382c01980SMichael Klier 64411fb5dc5SMichael Klier $avatar = ''; 64511fb5dc5SMichael Klier 64611fb5dc5SMichael Klier // check mail first 64711fb5dc5SMichael Klier if ($mail) { 64811fb5dc5SMichael Klier $avatar = $this->avatar->getXHTML($mail, $name, 'left'); 64911fb5dc5SMichael Klier } 65011fb5dc5SMichael Klier 65111fb5dc5SMichael Klier // okay maybe a user avatar? 65211fb5dc5SMichael Klier // FIXME make namespace configurable 65382c01980SMichael Klier $files = @glob(mediaFN('user') . '/' . $user . '.*'); 65411fb5dc5SMichael Klier if(!$avatar && $files) { 65511fb5dc5SMichael Klier $avatar = $this->avatar->getXHTML($user, $name, 'left'); 65682c01980SMichael Klier } 65711fb5dc5SMichael Klier 65811fb5dc5SMichael Klier // well it's a monster then 65911fb5dc5SMichael Klier if(!$avatar) { 66011fb5dc5SMichael Klier $avatar = $this->avatar->getXHTML($name, $name, 'left'); 66182c01980SMichael Klier } 66211fb5dc5SMichael Klier 66311fb5dc5SMichael Klier $head .= $avatar; 66411fb5dc5SMichael Klier 665f0fda08aSwikidesign } 666f0fda08aSwikidesign 6676046f25cSwikidesign if ($this->getConf('linkemail') && $mail) { 6684a0a1bd2Swikidesign $head .= $this->email($mail, $name, 'email fn'); 6696046f25cSwikidesign } elseif ($url) { 670ab7c9f30SMichael Klier $head .= $this->external_link($this->_checkURL($url), $name, 'urlextern url fn'); 671f0fda08aSwikidesign } else { 6724a0a1bd2Swikidesign $head .= '<span class="fn">'.$name.'</span>'; 673f0fda08aSwikidesign } 6744a0a1bd2Swikidesign if ($address) $head .= ', <span class="adr">'.$address.'</span>'; 6754a0a1bd2Swikidesign $head .= '</span>, '. 676f014bc86SMichael Klier '<abbr class="published" title="'.strftime('%Y-%m-%dT%H:%M:%SZ', $created).'">'. 677f014bc86SMichael Klier strftime($conf['dformat'], $created).'</abbr>'; 6784a0a1bd2Swikidesign if ($comment['edited']) $head .= ' (<abbr class="updated" title="'. 679f014bc86SMichael Klier strftime('%Y-%m-%dT%H:%M:%SZ', $modified).'">'.strftime($conf['dformat'], $modified). 6806046f25cSwikidesign '</abbr>)'; 681f014bc86SMichael Klier ptln($head, 8); 6824a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_head" 683f0fda08aSwikidesign 684f0fda08aSwikidesign // main comment content 6854a0a1bd2Swikidesign ptln('<div class="comment_body entry-content"'. 686f1c6610eSpierre.spring ($this->getConf('useavatar') ? $this->_get_style() : '').'>', 6); 68715cdad37Spierre.spring echo ($HIGH?html_hilight($comment['xhtml'],$HIGH):$comment['xhtml']).DOKU_LF; 6884a0a1bd2Swikidesign ptln('</div>', 6); // class="comment_body" 689f0fda08aSwikidesign 690f0fda08aSwikidesign if ($visible) { 6911184c36aSwikidesign ptln('<div class="comment_buttons">', 6); 692f0fda08aSwikidesign 693f0fda08aSwikidesign // show reply button? 694f1c4aa1aSwikidesign if (($data['status'] == 1) && !$reply && $comment['show'] 69531aab30eSGina Haeussge && ($this->getConf('allowguests') || $_SERVER['REMOTE_USER']) && $this->getConf('usethreading')) 6961e46d176Swikidesign $this->_button($cid, $this->getLang('btn_reply'), 'reply', true); 697f0fda08aSwikidesign 6981184c36aSwikidesign // show edit, show/hide and delete button? 6991184c36aSwikidesign if ((($user == $_SERVER['REMOTE_USER']) && ($user != '')) || (auth_ismanager())) { 7001e46d176Swikidesign $this->_button($cid, $lang['btn_secedit'], 'edit', true); 7011184c36aSwikidesign $label = ($comment['show'] ? $this->getLang('btn_hide') : $this->getLang('btn_show')); 7021184c36aSwikidesign $this->_button($cid, $label, 'toogle'); 7031184c36aSwikidesign $this->_button($cid, $lang['btn_delete'], 'delete'); 704f0fda08aSwikidesign } 7051184c36aSwikidesign ptln('</div>', 6); // class="comment_buttons" 7061184c36aSwikidesign } 7071184c36aSwikidesign ptln('</div>', 4); // class="hentry" 708f0fda08aSwikidesign } 709f0fda08aSwikidesign 710f1c6610eSpierre.spring function _print_form($cid, $reply) 711f1c6610eSpierre.spring { 71231aab30eSGina Haeussge if ($this->getConf('usethreading') && $reply == $cid) { 7134a0a1bd2Swikidesign ptln('<div class="comment_replies">', 4); 714f0fda08aSwikidesign $this->_form('', 'add', $cid); 7154a0a1bd2Swikidesign ptln('</div>', 4); // class="comment_replies" 716f0fda08aSwikidesign } 717f0fda08aSwikidesign } 718f0fda08aSwikidesign 719f1c6610eSpierre.spring function _print_replies($cid, &$data, $reply, &$visible) 720f1c6610eSpierre.spring { 721f1c6610eSpierre.spring $comment = $data['comments'][$cid]; 722f1c6610eSpierre.spring if (!count($comment['replies'])) { 723f1c6610eSpierre.spring return; 724f1c6610eSpierre.spring } 725f1c6610eSpierre.spring ptln('<div class="comment_replies"'.$this->_get_style().'>', 4); 726f1c6610eSpierre.spring $visible = ($comment['show'] && $visible); 727f1c6610eSpierre.spring foreach ($comment['replies'] as $rid) { 728f1c6610eSpierre.spring $this->_print($rid, $data, $cid, $reply, $visible); 729f1c6610eSpierre.spring } 730f1c6610eSpierre.spring ptln('</div>', 4); 731f1c6610eSpierre.spring } 732f1c6610eSpierre.spring 733f1c6610eSpierre.spring function _use_avatar() 734f1c6610eSpierre.spring { 735f1c6610eSpierre.spring if (is_null($this->use_avatar)) { 736f1c6610eSpierre.spring $this->use_avatar = $this->getConf('useavatar') 737f1c6610eSpierre.spring && (!plugin_isdisabled('avatar')) 738f1c6610eSpierre.spring && ($this->avatar =& plugin_load('helper', 'avatar')); 739f1c6610eSpierre.spring } 740f1c6610eSpierre.spring return $this->use_avatar; 741f1c6610eSpierre.spring } 742f1c6610eSpierre.spring 743f1c6610eSpierre.spring function _get_style() 744f1c6610eSpierre.spring { 745f1c6610eSpierre.spring if (is_null($this->style)){ 746f1c6610eSpierre.spring if ($this->_use_avatar()) { 747f1c6610eSpierre.spring $this->style = ' style="margin-left: '.($this->avatar->getConf('size') + 14).'px;"'; 748f1c6610eSpierre.spring } else { 749f1c6610eSpierre.spring $this->style = ' style="margin-left: 20px;"'; 750f1c6610eSpierre.spring } 751f1c6610eSpierre.spring } 752f1c6610eSpierre.spring return $this->style; 753f1c6610eSpierre.spring } 754f1c6610eSpierre.spring 755f0fda08aSwikidesign /** 756f0fda08aSwikidesign * Outputs the comment form 757f0fda08aSwikidesign */ 758f0fda08aSwikidesign function _form($raw = '', $act = 'add', $cid = NULL) { 7593011fb8bSMichael Klier global $lang; 7603011fb8bSMichael Klier global $conf; 7613011fb8bSMichael Klier global $ID; 7623011fb8bSMichael Klier global $INFO; 763f0fda08aSwikidesign 764f0fda08aSwikidesign // not for unregistered users when guest comments aren't allowed 765f0fda08aSwikidesign if (!$_SERVER['REMOTE_USER'] && !$this->getConf('allowguests')) return false; 766f0fda08aSwikidesign 7671ba72c23Swikidesign // fill $raw with $_REQUEST['text'] if it's empty (for failed CAPTCHA check) 7681ba72c23Swikidesign if (!$raw && ($_REQUEST['comment'] == 'show')) $raw = $_REQUEST['text']; 769f0fda08aSwikidesign ?> 7705ef1705fSiLoveiDo 771f0fda08aSwikidesign <div class="comment_form"> 772bea69264SMichael Klier <form id="discussion__comment_form" method="post" action="<?php echo script() ?>" accept-charset="<?php echo $lang['encoding'] ?>"> 773f0fda08aSwikidesign <div class="no"> 774f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 77561437513Swikidesign <input type="hidden" name="do" value="show" /> 776f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 777ed4846efSMichael Klier <input type="hidden" name="wikisyntaxok" id="discussion__comment_wikisyntaxok" value="<?php echo $this->getConf('wikisyntaxok') ?>" /> 778530693fbSMichael Klier <?php 779f0fda08aSwikidesign // for adding a comment 780f0fda08aSwikidesign if ($act == 'add') { 781f0fda08aSwikidesign ?> 782f0fda08aSwikidesign <input type="hidden" name="reply" value="<?php echo $cid ?>" /> 783f0fda08aSwikidesign <?php 78420c152acSMichael Klier // for guest/adminimport: show name, e-mail and subscribe to comments fields 78520c152acSMichael Klier if(!$_SERVER['REMOTE_USER'] or ($this->getConf('adminimport') && auth_ismanager())) { 786f0fda08aSwikidesign ?> 787f0fda08aSwikidesign <input type="hidden" name="user" value="<?php echo clientIP() ?>" /> 788f0fda08aSwikidesign <div class="comment_name"> 789f0fda08aSwikidesign <label class="block" for="discussion__comment_name"> 790f0fda08aSwikidesign <span><?php echo $lang['fullname'] ?>:</span> 79137e3c825SMichael 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'])?>" /> 792f0fda08aSwikidesign </label> 793f0fda08aSwikidesign </div> 794f0fda08aSwikidesign <div class="comment_mail"> 795f0fda08aSwikidesign <label class="block" for="discussion__comment_mail"> 796f0fda08aSwikidesign <span><?php echo $lang['email'] ?>:</span> 79737e3c825SMichael 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'])?>" /> 798f0fda08aSwikidesign </label> 799f0fda08aSwikidesign </div> 800f0fda08aSwikidesign <?php 801f0fda08aSwikidesign } 802f0fda08aSwikidesign 803f0fda08aSwikidesign // allow entering an URL 804f0fda08aSwikidesign if ($this->getConf('urlfield')) { 805f0fda08aSwikidesign ?> 806f0fda08aSwikidesign <div class="comment_url"> 807f0fda08aSwikidesign <label class="block" for="discussion__comment_url"> 808f0fda08aSwikidesign <span><?php echo $this->getLang('url') ?>:</span> 809e7c760b3Swikidesign <input type="text" class="edit" name="url" id="discussion__comment_url" size="50" tabindex="3" value="<?php echo hsc($_REQUEST['url'])?>" /> 810f0fda08aSwikidesign </label> 811f0fda08aSwikidesign </div> 812f0fda08aSwikidesign <?php 813f0fda08aSwikidesign } 814f0fda08aSwikidesign 815f0fda08aSwikidesign // allow entering an address 816f0fda08aSwikidesign if ($this->getConf('addressfield')) { 817f0fda08aSwikidesign ?> 818f0fda08aSwikidesign <div class="comment_address"> 819f0fda08aSwikidesign <label class="block" for="discussion__comment_address"> 820f0fda08aSwikidesign <span><?php echo $this->getLang('address') ?>:</span> 821e7c760b3Swikidesign <input type="text" class="edit" name="address" id="discussion__comment_address" size="50" tabindex="4" value="<?php echo hsc($_REQUEST['address'])?>" /> 822f0fda08aSwikidesign </label> 823f0fda08aSwikidesign </div> 824f0fda08aSwikidesign <?php 825f0fda08aSwikidesign } 826f0fda08aSwikidesign 827f0fda08aSwikidesign // allow setting the comment date 828a1ace3c9Swikidesign if ($this->getConf('adminimport') && (auth_ismanager())) { 829f0fda08aSwikidesign ?> 830f0fda08aSwikidesign <div class="comment_date"> 831f0fda08aSwikidesign <label class="block" for="discussion__comment_date"> 832f0fda08aSwikidesign <span><?php echo $this->getLang('date') ?>:</span> 833f0fda08aSwikidesign <input type="text" class="edit" name="date" id="discussion__comment_date" size="50" /> 834f0fda08aSwikidesign </label> 835f0fda08aSwikidesign </div> 836f0fda08aSwikidesign <?php 837f0fda08aSwikidesign } 838f0fda08aSwikidesign 839f0fda08aSwikidesign // for saving a comment 840f0fda08aSwikidesign } else { 841f0fda08aSwikidesign ?> 842f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 843f0fda08aSwikidesign <?php 844f0fda08aSwikidesign } 845f0fda08aSwikidesign ?> 846f0fda08aSwikidesign <div class="comment_text"> 8471de52da1SMichael Klier <div id="discussion__comment_toolbar"> 8481de52da1SMichael Klier <?php echo $this->getLang('entercomment')?> 8491de52da1SMichael Klier <?php if($this->getLang('wikisyntaxok')) echo ', ' . $this->getLang('wikisyntax') . ':';?> 8501de52da1SMichael Klier </div> 85137e3c825SMichael 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 85237e3c825SMichael Klier if($raw) { 85337e3c825SMichael Klier echo formText($raw); 85437e3c825SMichael Klier } else { 85537e3c825SMichael Klier echo $_REQUEST['text']; 85637e3c825SMichael Klier } 85737e3c825SMichael Klier ?></textarea> 858f0fda08aSwikidesign </div> 859e7c760b3Swikidesign <?php //bad and dirty event insert hook 860e7c760b3Swikidesign $evdata = array('writable' => true); 861e7c760b3Swikidesign trigger_event('HTML_EDITFORM_INJECTION', $evdata); 862e7c760b3Swikidesign ?> 863bea69264SMichael 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" /> 864ed4846efSMichael 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]" /> 8653011fb8bSMichael Klier 866189071d2SMichael Klier <?php if((!$_SERVER['REMOTE_USER'] || $_SERVER['REMOTE_USER'] && !$conf['subscribers']) && $this->getConf('subscribe')) { ?> 8673011fb8bSMichael Klier <div class="comment_subscribe"> 86811b1339eSGina Haeussge <input type="checkbox" id="discussion__comment_subscribe" name="subscribe" tabindex="6" /> 8693011fb8bSMichael Klier <label class="block" for="discussion__comment_subscribe"> 8703011fb8bSMichael Klier <span><?php echo $this->getLang('subscribe') ?></span> 8713011fb8bSMichael Klier </label> 8723011fb8bSMichael Klier </div> 8733011fb8bSMichael Klier <?php } ?> 8743011fb8bSMichael Klier 8753011fb8bSMichael Klier <div class="clearer"></div> 876ed4846efSMichael Klier <div id="discussion__comment_preview"> </div> 877f0fda08aSwikidesign </div> 878f0fda08aSwikidesign </form> 879f0fda08aSwikidesign </div> 880f0fda08aSwikidesign <?php 881f0fda08aSwikidesign if ($this->getConf('usecocomment')) echo $this->_coComment(); 882f0fda08aSwikidesign } 883f0fda08aSwikidesign 884f0fda08aSwikidesign /** 885f0fda08aSwikidesign * Adds a javascript to interact with coComments 886f0fda08aSwikidesign */ 887f0fda08aSwikidesign function _coComment() { 8882b18adb9SMichael Klier global $ID; 8892b18adb9SMichael Klier global $conf; 8902b18adb9SMichael Klier global $INFO; 891f0fda08aSwikidesign 892f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 893f0fda08aSwikidesign 894f0fda08aSwikidesign ?> 895f0fda08aSwikidesign <script type="text/javascript"><!--//--><![CDATA[//><!-- 896f0fda08aSwikidesign var blogTool = "DokuWiki"; 897f0fda08aSwikidesign var blogURL = "<?php echo DOKU_URL ?>"; 898f0fda08aSwikidesign var blogTitle = "<?php echo $conf['title'] ?>"; 899f0fda08aSwikidesign var postURL = "<?php echo wl($ID, '', true) ?>"; 900f0fda08aSwikidesign var postTitle = "<?php echo tpl_pagetitle($ID, true) ?>"; 901f0fda08aSwikidesign <?php 902f0fda08aSwikidesign if ($user) { 903f0fda08aSwikidesign ?> 904f0fda08aSwikidesign var commentAuthor = "<?php echo $INFO['userinfo']['name'] ?>"; 905f0fda08aSwikidesign <?php 906f0fda08aSwikidesign } else { 907f0fda08aSwikidesign ?> 908f0fda08aSwikidesign var commentAuthorFieldName = "name"; 909f0fda08aSwikidesign <?php 910f0fda08aSwikidesign } 911f0fda08aSwikidesign ?> 912f0fda08aSwikidesign var commentAuthorLoggedIn = <?php echo ($user ? 'true' : 'false') ?>; 913f0fda08aSwikidesign var commentFormID = "discussion__comment_form"; 914f0fda08aSwikidesign var commentTextFieldName = "text"; 915f0fda08aSwikidesign var commentButtonName = "submit"; 916f0fda08aSwikidesign var cocomment_force = false; 917f0fda08aSwikidesign //--><!]]></script> 918f0fda08aSwikidesign <script type="text/javascript" src="http://www.cocomment.com/js/cocomment.js"> 919f0fda08aSwikidesign </script> 920f0fda08aSwikidesign <?php 921f0fda08aSwikidesign } 922f0fda08aSwikidesign 923f0fda08aSwikidesign /** 924f0fda08aSwikidesign * General button function 925f0fda08aSwikidesign */ 9261e46d176Swikidesign function _button($cid, $label, $act, $jump = false) { 927f0fda08aSwikidesign global $ID; 9285ef1705fSiLoveiDo 9291e46d176Swikidesign $anchor = ($jump ? '#discussion__comment_form' : '' ); 930f0fda08aSwikidesign 931f0fda08aSwikidesign ?> 9326d1f4f20SGina Haeussge <form class="button discussion__<?php echo $act?>" method="get" action="<?php echo script().$anchor ?>"> 933f0fda08aSwikidesign <div class="no"> 934f0fda08aSwikidesign <input type="hidden" name="id" value="<?php echo $ID ?>" /> 93561437513Swikidesign <input type="hidden" name="do" value="show" /> 936f0fda08aSwikidesign <input type="hidden" name="comment" value="<?php echo $act ?>" /> 937f0fda08aSwikidesign <input type="hidden" name="cid" value="<?php echo $cid ?>" /> 938f0fda08aSwikidesign <input type="submit" value="<?php echo $label ?>" class="button" title="<?php echo $label ?>" /> 939f0fda08aSwikidesign </div> 940f0fda08aSwikidesign </form> 941f0fda08aSwikidesign <?php 942f0fda08aSwikidesign return true; 943f0fda08aSwikidesign } 944f0fda08aSwikidesign 945f0fda08aSwikidesign /** 946f0fda08aSwikidesign * Adds an entry to the comments changelog 947f0fda08aSwikidesign * 948f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 949f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 950f0fda08aSwikidesign */ 951f0fda08aSwikidesign function _addLogEntry($date, $id, $type = 'cc', $summary = '', $extra = '') { 952f0fda08aSwikidesign global $conf; 953f0fda08aSwikidesign 954f0fda08aSwikidesign $changelog = $conf['metadir'].'/_comments.changes'; 955f0fda08aSwikidesign 956f0fda08aSwikidesign if(!$date) $date = time(); //use current time if none supplied 957f0fda08aSwikidesign $remote = $_SERVER['REMOTE_ADDR']; 958f0fda08aSwikidesign $user = $_SERVER['REMOTE_USER']; 959f0fda08aSwikidesign 960f0fda08aSwikidesign $strip = array("\t", "\n"); 961f0fda08aSwikidesign $logline = array( 962f0fda08aSwikidesign 'date' => $date, 963f0fda08aSwikidesign 'ip' => $remote, 964f0fda08aSwikidesign 'type' => str_replace($strip, '', $type), 965f0fda08aSwikidesign 'id' => $id, 966f0fda08aSwikidesign 'user' => $user, 967f0fda08aSwikidesign 'sum' => str_replace($strip, '', $summary), 968f0fda08aSwikidesign 'extra' => str_replace($strip, '', $extra) 969f0fda08aSwikidesign ); 970f0fda08aSwikidesign 971f0fda08aSwikidesign // add changelog line 972f0fda08aSwikidesign $logline = implode("\t", $logline)."\n"; 973f0fda08aSwikidesign io_saveFile($changelog, $logline, true); //global changelog cache 974f0fda08aSwikidesign $this->_trimRecentCommentsLog($changelog); 97577a22ba2Swikidesign 97677a22ba2Swikidesign // tell the indexer to re-index the page 97777a22ba2Swikidesign @unlink(metaFN($id, '.indexed')); 978f0fda08aSwikidesign } 979f0fda08aSwikidesign 980f0fda08aSwikidesign /** 981f0fda08aSwikidesign * Trims the recent comments cache to the last $conf['changes_days'] recent 982f0fda08aSwikidesign * changes or $conf['recent'] items, which ever is larger. 983f0fda08aSwikidesign * The trimming is only done once a day. 984f0fda08aSwikidesign * 985f0fda08aSwikidesign * @author Ben Coburn <btcoburn@silicodon.net> 986f0fda08aSwikidesign */ 987f0fda08aSwikidesign function _trimRecentCommentsLog($changelog) { 988f0fda08aSwikidesign global $conf; 989f0fda08aSwikidesign 990f0fda08aSwikidesign if (@file_exists($changelog) && 991f0fda08aSwikidesign (filectime($changelog) + 86400) < time() && 992f0fda08aSwikidesign !@file_exists($changelog.'_tmp')) { 993f0fda08aSwikidesign 994f0fda08aSwikidesign io_lock($changelog); 995f0fda08aSwikidesign $lines = file($changelog); 996f0fda08aSwikidesign if (count($lines)<$conf['recent']) { 997f0fda08aSwikidesign // nothing to trim 998f0fda08aSwikidesign io_unlock($changelog); 999f0fda08aSwikidesign return true; 1000f0fda08aSwikidesign } 1001f0fda08aSwikidesign 1002f0fda08aSwikidesign io_saveFile($changelog.'_tmp', ''); // presave tmp as 2nd lock 1003f0fda08aSwikidesign $trim_time = time() - $conf['recent_days']*86400; 1004f0fda08aSwikidesign $out_lines = array(); 1005f0fda08aSwikidesign 1006e49085a2SMichael Klier $num = count($lines); 1007e49085a2SMichael Klier for ($i=0; $i<$num; $i++) { 1008f0fda08aSwikidesign $log = parseChangelogLine($lines[$i]); 1009f0fda08aSwikidesign if ($log === false) continue; // discard junk 1010f0fda08aSwikidesign if ($log['date'] < $trim_time) { 1011f0fda08aSwikidesign $old_lines[$log['date'].".$i"] = $lines[$i]; // keep old lines for now (append .$i to prevent key collisions) 1012f0fda08aSwikidesign } else { 1013f0fda08aSwikidesign $out_lines[$log['date'].".$i"] = $lines[$i]; // definitely keep these lines 1014f0fda08aSwikidesign } 1015f0fda08aSwikidesign } 1016f0fda08aSwikidesign 1017f0fda08aSwikidesign // sort the final result, it shouldn't be necessary, 1018f0fda08aSwikidesign // however the extra robustness in making the changelog cache self-correcting is worth it 1019f0fda08aSwikidesign ksort($out_lines); 1020f0fda08aSwikidesign $extra = $conf['recent'] - count($out_lines); // do we need extra lines do bring us up to minimum 1021f0fda08aSwikidesign if ($extra > 0) { 1022f0fda08aSwikidesign ksort($old_lines); 1023f0fda08aSwikidesign $out_lines = array_merge(array_slice($old_lines,-$extra),$out_lines); 1024f0fda08aSwikidesign } 1025f0fda08aSwikidesign 1026f0fda08aSwikidesign // save trimmed changelog 1027f0fda08aSwikidesign io_saveFile($changelog.'_tmp', implode('', $out_lines)); 1028f0fda08aSwikidesign @unlink($changelog); 1029f0fda08aSwikidesign if (!rename($changelog.'_tmp', $changelog)) { 1030f0fda08aSwikidesign // rename failed so try another way... 1031f0fda08aSwikidesign io_unlock($changelog); 1032f0fda08aSwikidesign io_saveFile($changelog, implode('', $out_lines)); 1033f0fda08aSwikidesign @unlink($changelog.'_tmp'); 1034f0fda08aSwikidesign } else { 1035f0fda08aSwikidesign io_unlock($changelog); 1036f0fda08aSwikidesign } 1037f0fda08aSwikidesign return true; 1038f0fda08aSwikidesign } 1039f0fda08aSwikidesign } 1040f0fda08aSwikidesign 1041f0fda08aSwikidesign /** 1042f0fda08aSwikidesign * Sends a notify mail on new comment 1043f0fda08aSwikidesign * 1044f0fda08aSwikidesign * @param array $comment data array of the new comment 1045f0fda08aSwikidesign * 1046f0fda08aSwikidesign * @author Andreas Gohr <andi@splitbrain.org> 1047f0fda08aSwikidesign * @author Esther Brunner <wikidesign@gmail.com> 1048f0fda08aSwikidesign */ 10499881d835SMichael Klier function _notify($comment, &$subscribers) { 1050f0fda08aSwikidesign global $conf; 1051f0fda08aSwikidesign global $ID; 1052f0fda08aSwikidesign 10539881d835SMichael Klier $notify_text = io_readfile($this->localfn('subscribermail')); 10549881d835SMichael Klier $confirm_text = io_readfile($this->localfn('confirmsubscribe')); 10559881d835SMichael Klier $subject_notify = '['.$conf['title'].'] '.$this->getLang('mail_newcomment'); 10569881d835SMichael Klier $subject_subscribe = '['.$conf['title'].'] '.$this->getLang('subscribe'); 1057f0fda08aSwikidesign 10586046f25cSwikidesign $search = array( 10596046f25cSwikidesign '@PAGE@', 10606046f25cSwikidesign '@TITLE@', 10616046f25cSwikidesign '@DATE@', 10626046f25cSwikidesign '@NAME@', 10636046f25cSwikidesign '@TEXT@', 10648b42cefbSMichael Klier '@COMMENTURL@', 10656046f25cSwikidesign '@UNSUBSCRIBE@', 10666046f25cSwikidesign '@DOKUWIKIURL@', 10676046f25cSwikidesign ); 10683011fb8bSMichael Klier 10693011fb8bSMichael Klier // notify page subscribers 10708c6e87cfSMichael Klier if ($conf['subscribers'] || $conf['notify']) { 1071189071d2SMichael Klier $list = explode(',', subscriber_addresslist($ID)); 10728c6e87cfSMichael Klier $to = (!empty($conf['notify'])) ? $conf['notify'] : array_pop($list); 1073189071d2SMichael Klier $bcc = implode(',', $list); 10743011fb8bSMichael Klier 10756046f25cSwikidesign $replace = array( 10766046f25cSwikidesign $ID, 10776046f25cSwikidesign $conf['title'], 1078f014bc86SMichael Klier strftime($conf['dformat'], $comment['date']['created']), 10796046f25cSwikidesign $comment['user']['name'], 10806046f25cSwikidesign $comment['raw'], 10811810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 10826046f25cSwikidesign wl($ID, 'do=unsubscribe', true, '&'), 10836046f25cSwikidesign DOKU_URL, 10846046f25cSwikidesign ); 1085f0fda08aSwikidesign 10869881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 10879881d835SMichael Klier mail_send($to, $subject_notify, $body, $conf['mailfrom'], '', $bcc); 10883011fb8bSMichael Klier } 1089f0fda08aSwikidesign 10903011fb8bSMichael Klier // notify comment subscribers 10913011fb8bSMichael Klier if (!empty($subscribers)) { 10923011fb8bSMichael Klier 10939881d835SMichael Klier foreach($subscribers as $mail => $data) { 10943011fb8bSMichael Klier $to = $mail; 10953011fb8bSMichael Klier 10969881d835SMichael Klier if($data['active']) { 10973011fb8bSMichael Klier $replace = array( 10983011fb8bSMichael Klier $ID, 10993011fb8bSMichael Klier $conf['title'], 11003011fb8bSMichael Klier strftime($conf['dformat'], $comment['date']['created']), 11013011fb8bSMichael Klier $comment['user']['name'], 11023011fb8bSMichael Klier $comment['raw'], 11031810ba9cSMichael Klier wl($ID, '', true) . '#comment_' . $comment['cid'], 11049881d835SMichael Klier wl($ID, 'do=unsubscribe&hash=' . $data['hash'], true, '&'), 11053011fb8bSMichael Klier DOKU_URL, 11063011fb8bSMichael Klier ); 11073011fb8bSMichael Klier 11089881d835SMichael Klier $body = str_replace($search, $replace, $notify_text); 11099881d835SMichael Klier mail_send($to, $subject_notify, $body, $conf['mailfrom']); 11109881d835SMichael Klier } elseif(!$data['active'] && !$data['confirmsent']) { 11119881d835SMichael Klier $search = array( 11129881d835SMichael Klier '@PAGE@', 11139881d835SMichael Klier '@TITLE@', 11149881d835SMichael Klier '@SUBSCRIBE@', 11159881d835SMichael Klier '@DOKUWIKIURL@', 11169881d835SMichael Klier ); 11179881d835SMichael Klier $replace = array( 11189881d835SMichael Klier $ID, 11199881d835SMichael Klier $conf['title'], 11209881d835SMichael Klier wl($ID, 'do=confirmsubscribe&hash=' . $data['hash'], true, '&'), 11219881d835SMichael Klier DOKU_URL, 11229881d835SMichael Klier ); 11239881d835SMichael Klier 11249881d835SMichael Klier $body = str_replace($search, $replace, $confirm_text); 11259881d835SMichael Klier mail_send($to, $subject_subscribe, $body, $conf['mailfrom']); 11269881d835SMichael Klier $subscribers[$mail]['confirmsent'] = true; 11279881d835SMichael Klier } 11283011fb8bSMichael Klier } 11293011fb8bSMichael Klier } 1130f0fda08aSwikidesign } 1131f0fda08aSwikidesign 1132f0fda08aSwikidesign /** 1133f0fda08aSwikidesign * Counts the number of visible comments 1134f0fda08aSwikidesign */ 1135f0fda08aSwikidesign function _count($data) { 1136f0fda08aSwikidesign $number = 0; 1137f0fda08aSwikidesign foreach ($data['comments'] as $cid => $comment) { 1138f0fda08aSwikidesign if ($comment['parent']) continue; 1139f0fda08aSwikidesign if (!$comment['show']) continue; 1140f0fda08aSwikidesign $number++; 1141f0fda08aSwikidesign $rids = $comment['replies']; 1142f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1143f0fda08aSwikidesign } 1144f0fda08aSwikidesign return $number; 1145f0fda08aSwikidesign } 1146f0fda08aSwikidesign 1147f0fda08aSwikidesign function _countReplies(&$data, $rids) { 1148f0fda08aSwikidesign $number = 0; 1149f0fda08aSwikidesign foreach ($rids as $rid) { 11502ee3dca3Swikidesign if (!isset($data['comments'][$rid])) continue; // reply was removed 1151f0fda08aSwikidesign if (!$data['comments'][$rid]['show']) continue; 1152f0fda08aSwikidesign $number++; 1153f0fda08aSwikidesign $rids = $data['comments'][$rid]['replies']; 1154f0fda08aSwikidesign if (count($rids)) $number = $number + $this->_countReplies($data, $rids); 1155f0fda08aSwikidesign } 1156f0fda08aSwikidesign return $number; 1157f0fda08aSwikidesign } 1158f0fda08aSwikidesign 1159f0fda08aSwikidesign /** 1160f0fda08aSwikidesign * Renders the comment text 1161f0fda08aSwikidesign */ 1162f0fda08aSwikidesign function _render($raw) { 1163f0fda08aSwikidesign if ($this->getConf('wikisyntaxok')) { 1164f0fda08aSwikidesign $xhtml = $this->render($raw); 1165f0fda08aSwikidesign } else { // wiki syntax not allowed -> just encode special chars 1166f0fda08aSwikidesign $xhtml = htmlspecialchars(trim($raw)); 1167f0fda08aSwikidesign } 1168f0fda08aSwikidesign return $xhtml; 1169f0fda08aSwikidesign } 1170f0fda08aSwikidesign 1171f0fda08aSwikidesign /** 1172479dd10fSwikidesign * Finds out whether there is a discussion section for the current page 1173479dd10fSwikidesign */ 1174e6a89be2Swikidesign function _hasDiscussion(&$title) { 1175b2ac3b3bSwikidesign global $ID; 11764a0a1bd2Swikidesign 1177479dd10fSwikidesign $cfile = metaFN($ID, '.comments'); 1178479dd10fSwikidesign 1179479dd10fSwikidesign if (!@file_exists($cfile)) { 11802b18adb9SMichael Klier if ($this->getConf('automatic')) { 11812b18adb9SMichael Klier return true; 11822b18adb9SMichael Klier } else { 11832b18adb9SMichael Klier return false; 11842b18adb9SMichael Klier } 1185479dd10fSwikidesign } 1186479dd10fSwikidesign 1187479dd10fSwikidesign $comments = unserialize(io_readFile($cfile, false)); 1188479dd10fSwikidesign 118907c376bbSwikidesign if ($comments['title']) $title = hsc($comments['title']); 1190479dd10fSwikidesign $num = $comments['number']; 1191479dd10fSwikidesign if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return false; 1192479dd10fSwikidesign else return true; 1193479dd10fSwikidesign } 1194479dd10fSwikidesign 1195479dd10fSwikidesign /** 1196e7c760b3Swikidesign * Creates a new thread page 1197e7c760b3Swikidesign */ 11986046f25cSwikidesign function _newThread() { 119961437513Swikidesign global $ID, $INFO; 1200f0fda08aSwikidesign 12011ea794e5Swikidesign $ns = cleanID($_REQUEST['ns']); 1202f0fda08aSwikidesign $title = str_replace(':', '', $_REQUEST['title']); 12032e80cd5fSwikidesign $back = $ID; 12042e80cd5fSwikidesign $ID = ($ns ? $ns.':' : '').cleanID($title); 12052e80cd5fSwikidesign $INFO = pageinfo(); 1206f0fda08aSwikidesign 1207f0fda08aSwikidesign // check if we are allowed to create this file 12082e80cd5fSwikidesign if ($INFO['perm'] >= AUTH_CREATE) { 1209f0fda08aSwikidesign 1210f0fda08aSwikidesign //check if locked by anyone - if not lock for my self 12112e80cd5fSwikidesign if ($INFO['locked']) return 'locked'; 12122e80cd5fSwikidesign else lock($ID); 1213f0fda08aSwikidesign 1214f0fda08aSwikidesign // prepare the new thread file with default stuff 12152e80cd5fSwikidesign if (!@file_exists($INFO['filepath'])) { 1216f0fda08aSwikidesign global $TEXT; 1217f0fda08aSwikidesign 12182e80cd5fSwikidesign $TEXT = pageTemplate(array(($ns ? $ns.':' : '').$title)); 12191433886fSwikidesign if (!$TEXT) { 122061437513Swikidesign $data = array('id' => $ID, 'ns' => $ns, 'title' => $title, 'back' => $back); 122161437513Swikidesign $TEXT = $this->_pageTemplate($data); 12222e80cd5fSwikidesign } 12232e80cd5fSwikidesign return 'preview'; 1224f0fda08aSwikidesign } else { 12252e80cd5fSwikidesign return 'edit'; 1226f0fda08aSwikidesign } 1227f0fda08aSwikidesign } else { 12282e80cd5fSwikidesign return 'show'; 1229f0fda08aSwikidesign } 1230f0fda08aSwikidesign } 1231f0fda08aSwikidesign 1232e7c760b3Swikidesign /** 123361437513Swikidesign * Adapted version of pageTemplate() function 123461437513Swikidesign */ 123561437513Swikidesign function _pageTemplate($data) { 123661437513Swikidesign global $conf, $INFO; 123761437513Swikidesign 123861437513Swikidesign $id = $data['id']; 123961437513Swikidesign $user = $_SERVER['REMOTE_USER']; 124061437513Swikidesign $tpl = io_readFile(DOKU_PLUGIN.'discussion/_template.txt'); 124161437513Swikidesign 124261437513Swikidesign // standard replacements 124361437513Swikidesign $replace = array( 124461437513Swikidesign '@NS@' => $data['ns'], 124561437513Swikidesign '@PAGE@' => strtr(noNS($id),'_',' '), 124661437513Swikidesign '@USER@' => $user, 124761437513Swikidesign '@NAME@' => $INFO['userinfo']['name'], 124861437513Swikidesign '@MAIL@' => $INFO['userinfo']['mail'], 1249f014bc86SMichael Klier '@DATE@' => strftime($conf['dformat']), 125061437513Swikidesign ); 125161437513Swikidesign 125261437513Swikidesign // additional replacements 125361437513Swikidesign $replace['@BACK@'] = $data['back']; 125461437513Swikidesign $replace['@TITLE@'] = $data['title']; 125561437513Swikidesign 125661437513Swikidesign // avatar if useavatar and avatar plugin available 125761437513Swikidesign if ($this->getConf('useavatar') 125861437513Swikidesign && (@file_exists(DOKU_PLUGIN.'avatar/syntax.php')) 125961437513Swikidesign && (!plugin_isdisabled('avatar'))) { 126061437513Swikidesign $replace['@AVATAR@'] = '{{avatar>'.$user.' }} '; 126161437513Swikidesign } else { 126261437513Swikidesign $replace['@AVATAR@'] = ''; 126361437513Swikidesign } 126461437513Swikidesign 126561437513Swikidesign // tag if tag plugin is available 126661437513Swikidesign if ((@file_exists(DOKU_PLUGIN.'tag/syntax/tag.php')) 126761437513Swikidesign && (!plugin_isdisabled('tag'))) { 126861437513Swikidesign $replace['@TAG@'] = "\n\n{{tag>}}"; 126961437513Swikidesign } else { 127061437513Swikidesign $replace['@TAG@'] = ''; 127161437513Swikidesign } 127261437513Swikidesign 127361437513Swikidesign // do the replace 127461437513Swikidesign $tpl = str_replace(array_keys($replace), array_values($replace), $tpl); 127561437513Swikidesign return $tpl; 127661437513Swikidesign } 127761437513Swikidesign 127861437513Swikidesign /** 1279e7c760b3Swikidesign * Checks if the CAPTCHA string submitted is valid 1280e7c760b3Swikidesign * 1281e7c760b3Swikidesign * @author Andreas Gohr <gohr@cosmocode.de> 1282e7c760b3Swikidesign * @adaption Esther Brunner <wikidesign@gmail.com> 1283e7c760b3Swikidesign */ 12846046f25cSwikidesign function _captchaCheck() { 1285be1d1f2fSGina Haeussge if (plugin_isdisabled('captcha') || (!$captcha = plugin_load('helper', 'captcha'))) 1286be1d1f2fSGina Haeussge return; // CAPTCHA is disabled or not available 1287e7c760b3Swikidesign 1288d1c29589Swikidesign // do nothing if logged in user and no CAPTCHA required 1289d1c29589Swikidesign if (!$captcha->getConf('forusers') && $_SERVER['REMOTE_USER']) return; 1290d1c29589Swikidesign 1291e7c760b3Swikidesign // compare provided string with decrypted captcha 1292e7c760b3Swikidesign $rand = PMA_blowfish_decrypt($_REQUEST['plugin__captcha_secret'], auth_cookiesalt()); 1293e7c760b3Swikidesign $code = $captcha->_generateCAPTCHA($captcha->_fixedIdent(), $rand); 1294e7c760b3Swikidesign 1295e7c760b3Swikidesign if (!$_REQUEST['plugin__captcha_secret'] || 1296e7c760b3Swikidesign !$_REQUEST['plugin__captcha'] || 1297e7c760b3Swikidesign strtoupper($_REQUEST['plugin__captcha']) != $code) { 1298e7c760b3Swikidesign 1299e7c760b3Swikidesign // CAPTCHA test failed! Continue to edit instead of saving 1300e7c760b3Swikidesign msg($captcha->getLang('testfailed'), -1); 1301e7c760b3Swikidesign if ($_REQUEST['comment'] == 'save') $_REQUEST['comment'] = 'edit'; 1302e7c760b3Swikidesign elseif ($_REQUEST['comment'] == 'add') $_REQUEST['comment'] = 'show'; 1303e7c760b3Swikidesign } 1304e7c760b3Swikidesign // if we arrive here it was a valid save 1305e7c760b3Swikidesign } 1306e7c760b3Swikidesign 1307a1ca9e44Swikidesign /** 1308a1ca9e44Swikidesign * Adds the comments to the index 1309a1ca9e44Swikidesign */ 131061437513Swikidesign function idx_add_discussion(&$event, $param) { 1311a1ca9e44Swikidesign 1312a1ca9e44Swikidesign // get .comments meta file name 1313a1ca9e44Swikidesign $file = metaFN($event->data[0], '.comments'); 1314a1ca9e44Swikidesign 1315a1ca9e44Swikidesign if (@file_exists($file)) $data = unserialize(io_readFile($file, false)); 1316a1ca9e44Swikidesign if ((!$data['status']) || ($data['number'] == 0)) return; // comments are turned off 1317a1ca9e44Swikidesign 1318a1ca9e44Swikidesign // now add the comments 1319a1ca9e44Swikidesign if (isset($data['comments'])) { 1320a1ca9e44Swikidesign foreach ($data['comments'] as $key => $value) { 132177a22ba2Swikidesign $event->data[1] .= $this->_addCommentWords($key, $data); 1322a1ca9e44Swikidesign } 1323a1ca9e44Swikidesign } 1324a1ca9e44Swikidesign } 1325a1ca9e44Swikidesign 1326a1ca9e44Swikidesign /** 1327a1ca9e44Swikidesign * Adds the words of a given comment to the index 1328a1ca9e44Swikidesign */ 1329efbe59d0Swikidesign function _addCommentWords($cid, &$data, $parent = '') { 1330a1ca9e44Swikidesign 1331efbe59d0Swikidesign if (!isset($data['comments'][$cid])) return ''; // comment was removed 1332a1ca9e44Swikidesign $comment = $data['comments'][$cid]; 1333a1ca9e44Swikidesign 1334efbe59d0Swikidesign if (!is_array($comment)) return ''; // corrupt datatype 1335efbe59d0Swikidesign if ($comment['parent'] != $parent) return ''; // reply to an other comment 1336efbe59d0Swikidesign if (!$comment['show']) return ''; // hidden comment 1337a1ca9e44Swikidesign 1338efbe59d0Swikidesign $text = $comment['raw']; // we only add the raw comment text 1339efbe59d0Swikidesign if (is_array($comment['replies'])) { // and the replies 1340efbe59d0Swikidesign foreach ($comment['replies'] as $rid) { 1341efbe59d0Swikidesign $text .= $this->_addCommentWords($rid, $data, $cid); 1342a1ca9e44Swikidesign } 1343a1ca9e44Swikidesign } 1344efbe59d0Swikidesign return ' '.$text; 1345efbe59d0Swikidesign } 1346a10b5c98SMichael Klier 1347a10b5c98SMichael Klier /** 1348a10b5c98SMichael Klier * Only allow http(s) URLs and append http:// to URLs if needed 1349a10b5c98SMichael Klier */ 1350a10b5c98SMichael Klier function _checkURL($url) { 1351a10b5c98SMichael Klier if(preg_match("#^http://|^https://#", $url)) { 1352a10b5c98SMichael Klier return hsc($url); 1353a10b5c98SMichael Klier } elseif(substr($url, 0, 4) == 'www.') { 1354a10b5c98SMichael Klier return hsc('http://' . $url); 1355a10b5c98SMichael Klier } else { 1356a10b5c98SMichael Klier return ''; 1357a10b5c98SMichael Klier } 1358a10b5c98SMichael Klier } 1359f0fda08aSwikidesign} 136031aab30eSGina Haeussge 136131aab30eSGina Haeussgefunction _sortCallback($a, $b) { 136231aab30eSGina Haeussge if (is_array($a['date'])) { // new format 136331aab30eSGina Haeussge $createdA = $a['date']['created']; 136431aab30eSGina Haeussge } else { // old format 136531aab30eSGina Haeussge $createdA = $a['date']; 136631aab30eSGina Haeussge } 136731aab30eSGina Haeussge 136831aab30eSGina Haeussge if (is_array($b['date'])) { // new format 136931aab30eSGina Haeussge $createdB = $b['date']['created']; 137031aab30eSGina Haeussge } else { // old format 137131aab30eSGina Haeussge $createdB = $b['date']; 137231aab30eSGina Haeussge } 137331aab30eSGina Haeussge 137431aab30eSGina Haeussge if ($createdA == $createdB) 137531aab30eSGina Haeussge return 0; 137631aab30eSGina Haeussge else 137731aab30eSGina Haeussge return ($createdA < $createdB) ? -1 : 1; 137831aab30eSGina Haeussge} 137931aab30eSGina Haeussge 1380530693fbSMichael Klier// vim:ts=4:sw=4:et:enc=utf-8: 1381