', 10);
ptln('
', 10);
ptln('
', 10);
echo html_buildlist($comments, 'admin_discussion', array($this, '_commentItem'), array($this, '_li_comment'));
$this->_actionButtons($thread['id']);
}
$this->_browseDiscussionLinks($more, $first, $num);
}
/**
* Returns an array of pages with discussion sections, sorted by recent comments
*
* @return array
*/
function _getThreads() {
global $conf;
require_once(DOKU_INC.'inc/search.php');
// returns the list of pages in the given namespace and it's subspaces
$items = array();
search($items, $conf['datadir'], 'search_allpages', array());
// add pages with comments to result
$result = array();
foreach ($items as $item) {
$id = $item['id'];
// some checks
$file = metaFN($id, '.comments');
if (!@file_exists($file)) continue; // skip if no comments file
$date = filemtime($file);
$result[] = array(
'id' => $id,
'file' => $file,
'date' => $date,
);
}
// finally sort by time of last comment
usort($result, array('admin_plugin_discussion', '_threadCmp'));
return $result;
}
/**
* Callback for comparison of thread data.
*
* Used for sorting threads in descending order by date of last comment.
* If this date happens to be equal for the compared threads, page id
* is used as second comparison attribute.
*
* @param array $a
* @param array $b
* @return int
*/
function _threadCmp($a, $b) {
if ($a['date'] == $b['date']) {
return strcmp($a['id'], $b['id']);
}
return ($a['date'] < $b['date']) ? 1 : -1;
}
/**
* Outputs header, page ID and status of a discussion thread
*
* @param array $thread
* @return bool
*/
function _threadHead($thread) {
$id = $thread['id'];
$labels = array(
0 => $this->getLang('off'),
1 => $this->getLang('open'),
2 => $this->getLang('closed')
);
$title = p_get_metadata($id, 'title');
if (!$title) {
$title = $id;
}
ptln('
'.hsc($title).'
', 6);
ptln('
', 6);
ptln('
', 6);
ptln('
'.$id.' ', 8);
return true;
}
/**
* Returns the full comments data for a given wiki page
*
* @param array $thread
* @return array|bool
*/
function _getComments(&$thread) {
$id = $thread['id'];
if (!$thread['file']) {
$thread['file'] = metaFN($id, '.comments');
}
if (!@file_exists($thread['file'])) return false; // no discussion thread at all
$data = unserialize(io_readFile($thread['file'], false));
$thread['status'] = $data['status'];
$thread['number'] = $data['number'];
if (!$data['status']) return false; // comments are turned off
if (!$data['comments']) return false; // no comments
$result = array();
foreach ($data['comments'] as $cid => $comment) {
$this->_addComment($cid, $data, $result);
}
if (empty($result)) {
return false;
} else {
return $result;
}
}
/**
* Recursive function to add the comment hierarchy to the result
*
* @param string $cid
* @param array $data
* @param array $result
* @param string $parent
* @param int $level
*/
function _addComment($cid, &$data, &$result, $parent = '', $level = 1) {
if (!is_array($data['comments'][$cid])) return; // corrupt datatype
$comment = $data['comments'][$cid];
if ($comment['parent'] != $parent) return; // answer to another comment
// okay, add the comment to the result
$comment['id'] = $cid;
$comment['level'] = $level;
$result[] = $comment;
// check answers to this comment
if (count($comment['replies'])) {
foreach ($comment['replies'] as $rid) {
$this->_addComment($rid, $data, $result, $cid, $level + 1);
}
}
}
/**
* Checkbox and info about a comment item
*
* @param array $comment
* @return string
*/
function _commentItem($comment) {
global $conf;
// prepare variables
if (is_array($comment['user'])) { // new format
$name = $comment['user']['name'];
$mail = $comment['user']['mail'];
} else { // old format
$name = $comment['name'];
$mail = $comment['mail'];
}
if (is_array($comment['date'])) { // new format
$created = $comment['date']['created'];
} else { // old format
$created = $comment['date'];
}
$abstract = preg_replace('/\s+?/', ' ', strip_tags($comment['xhtml']));
if (utf8_strlen($abstract) > 160) {
$abstract = utf8_substr($abstract, 0, 160).'...';
}
return '
'.
$this->email($mail, $name, 'email').', '.strftime($conf['dformat'], $created).': '.
'
'.$abstract.'';
}
/**
* list item tag
*
* @param array $comment
* @return string
*/
function _li_comment($comment) {
$show = ($comment['show'] ? '' : ' hidden');
return '
', 10); // class="no"
ptln('', 8);
ptln('
', 6); // class="level2"
return true;
}
/**
* Displays links to older newer discussions
*
* @param bool $more
* @param int $first
* @param int $num
* @return bool
*/
function _browseDiscussionLinks($more, $first, $num) {
global $ID;
if (($first == 0) && (!$more)) return true;
$params = array('do' => 'admin', 'page' => 'discussion');
$last = $first+$num;
ptln('', 8);
$ret = '';
if ($first > 0) {
$first -= $num;
if ($first < 0) {
$first = 0;
}
$params['first'] = $first;
ptln('
', 8);
$ret = '<< '.$this->getLang('newer').'';
if ($more) {
$ret .= ' | ';
} else {
ptln($ret, 10);
ptln('
', 8);
}
} else if ($more) {
ptln('
', 8);
}
if ($more) {
$params['first'] = $last;
$ret .= ''.$this->getLang('older').' >>';
ptln($ret, 10);
ptln('
', 8);
}
ptln('
', 6); // class="level1"
return true;
}
/**
* Changes the status of a comment
*
* @param $new
* @return bool
*/
function _changeStatus($new) {
global $ID;
// get discussion meta file name
$file = metaFN($ID, '.comments');
$data = unserialize(io_readFile($file, false));
$old = $data['status'];
if ($old == $new) return true;
// save the comment metadata file
$data['status'] = $new;
io_saveFile($file, serialize($data));
// look for ~~DISCUSSION~~ command in page file and change it accordingly
$patterns = array('~~DISCUSSION:off\2~~', '~~DISCUSSION\2~~', '~~DISCUSSION:closed\2~~');
$replace = $patterns[$new];
$wiki = preg_replace('/~~DISCUSSION([\w:]*)(\|?.*?)~~/', $replace, rawWiki($ID));
saveWikiText($ID, $wiki, $this->getLang('statuschanged'), true);
return true;
}
}
// vim:ts=4:sw=4:et:enc=utf-8: