xref: /plugin/discussion/helper.php (revision 5fc512fb4e57f6c46a139cbe38495abc4108b731)
1*5fc512fbSwikidesign<?php
2*5fc512fbSwikidesign/**
3*5fc512fbSwikidesign * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
4*5fc512fbSwikidesign * @author     Esther Brunner <wikidesign@gmail.com>
5*5fc512fbSwikidesign */
6*5fc512fbSwikidesign
7*5fc512fbSwikidesign// must be run within Dokuwiki
8*5fc512fbSwikidesignif (!defined('DOKU_INC')) die();
9*5fc512fbSwikidesign
10*5fc512fbSwikidesignif (!defined('DOKU_LF')) define('DOKU_LF', "\n");
11*5fc512fbSwikidesignif (!defined('DOKU_TAB')) define('DOKU_TAB', "\t");
12*5fc512fbSwikidesign
13*5fc512fbSwikidesignclass helper_plugin_discussion extends DokuWiki_Plugin {
14*5fc512fbSwikidesign
15*5fc512fbSwikidesign  function getInfo(){
16*5fc512fbSwikidesign    return array(
17*5fc512fbSwikidesign      'author' => 'Esther Brunner',
18*5fc512fbSwikidesign      'email'  => 'wikidesign@gmail.com',
19*5fc512fbSwikidesign      'date'   => '2006-12-10',
20*5fc512fbSwikidesign      'name'   => 'Discussion Plugin (helper class)',
21*5fc512fbSwikidesign      'desc'   => 'Functions to get info about comments to a wiki page',
22*5fc512fbSwikidesign      'url'    => 'http://www.wikidesign/en/plugin/discussion/start',
23*5fc512fbSwikidesign    );
24*5fc512fbSwikidesign  }
25*5fc512fbSwikidesign
26*5fc512fbSwikidesign  function getMethods(){
27*5fc512fbSwikidesign    $result = array();
28*5fc512fbSwikidesign    $result[] = array(
29*5fc512fbSwikidesign      'name'   => 'th',
30*5fc512fbSwikidesign      'desc'   => 'returns the header of the comments column for pagelist',
31*5fc512fbSwikidesign      'return' => array('header' => 'string'),
32*5fc512fbSwikidesign    );
33*5fc512fbSwikidesign    $result[] = array(
34*5fc512fbSwikidesign      'name'   => 'td',
35*5fc512fbSwikidesign      'desc'   => 'returns the link to the discussion section with number of comments',
36*5fc512fbSwikidesign      'params' => array(
37*5fc512fbSwikidesign        'id' => 'string',
38*5fc512fbSwikidesign        'number of comments (optional)' => 'integer'),
39*5fc512fbSwikidesign      'return' => array('link' => 'string'),
40*5fc512fbSwikidesign    );
41*5fc512fbSwikidesign    $result[] = array(
42*5fc512fbSwikidesign      'name'   => 'getThreads',
43*5fc512fbSwikidesign      'desc'   => 'returns pages with discussion sections, sorted by recent comments',
44*5fc512fbSwikidesign      'params' => array(
45*5fc512fbSwikidesign        'namespace' => 'string',
46*5fc512fbSwikidesign        'number (optional)' => 'integer'),
47*5fc512fbSwikidesign      'return' => array('pages' => 'array'),
48*5fc512fbSwikidesign    );
49*5fc512fbSwikidesign    $result[] = array(
50*5fc512fbSwikidesign      'name'   => 'getComments',
51*5fc512fbSwikidesign      'desc'   => 'returns recently added or edited comments individually',
52*5fc512fbSwikidesign      'params' => array(
53*5fc512fbSwikidesign        'namespace' => 'string',
54*5fc512fbSwikidesign        'number (optional)' => 'integer'),
55*5fc512fbSwikidesign      'return' => array('pages' => 'array'),
56*5fc512fbSwikidesign    );
57*5fc512fbSwikidesign    return $result;
58*5fc512fbSwikidesign  }
59*5fc512fbSwikidesign
60*5fc512fbSwikidesign  /**
61*5fc512fbSwikidesign   * Returns the column header for the Pagelist Plugin
62*5fc512fbSwikidesign   */
63*5fc512fbSwikidesign  function th(){
64*5fc512fbSwikidesign    return $this->getLang('discussion');
65*5fc512fbSwikidesign  }
66*5fc512fbSwikidesign
67*5fc512fbSwikidesign  /**
68*5fc512fbSwikidesign   * Returns the link to the discussion section of a page
69*5fc512fbSwikidesign   */
70*5fc512fbSwikidesign  function td($id, $num = NULL){
71*5fc512fbSwikidesign    $discuss = $id.'#'.cleanID($this->getLang('discussion'));
72*5fc512fbSwikidesign
73*5fc512fbSwikidesign    if (!isset($num)){
74*5fc512fbSwikidesign      $cfile = metaFN($id, '.comments');
75*5fc512fbSwikidesign      $comments = unserialize(io_readFile($cfile, false));
76*5fc512fbSwikidesign
77*5fc512fbSwikidesign      $num = $comments['number'];
78*5fc512fbSwikidesign      if ((!$comments['status']) || (($comments['status'] == 2) && (!$num))) return '';
79*5fc512fbSwikidesign    }
80*5fc512fbSwikidesign
81*5fc512fbSwikidesign    if ($num == 0) $comment = '0 '.$this->getLang('comments');
82*5fc512fbSwikidesign    elseif ($num == 1) $comment = '1 '.$this->getLang('comment');
83*5fc512fbSwikidesign    else $comment = $num.' '.$this->getLang('comments');
84*5fc512fbSwikidesign
85*5fc512fbSwikidesign    return '<a href="'.wl($discuss).'" class="wikilink1" title="'.$discuss.'">'.
86*5fc512fbSwikidesign      $comment.'</a>';
87*5fc512fbSwikidesign  }
88*5fc512fbSwikidesign
89*5fc512fbSwikidesign  /**
90*5fc512fbSwikidesign   * Returns an array of pages with discussion sections, sorted by recent comments
91*5fc512fbSwikidesign   */
92*5fc512fbSwikidesign  function getThreads($ns, $num = NULL){
93*5fc512fbSwikidesign    global $conf;
94*5fc512fbSwikidesign
95*5fc512fbSwikidesign    require_once(DOKU_INC.'inc/search.php');
96*5fc512fbSwikidesign
97*5fc512fbSwikidesign    $dir = $conf['datadir'].($ns ? '/'.str_replace(':', '/', $ns): '');
98*5fc512fbSwikidesign
99*5fc512fbSwikidesign    // returns the list of pages in the given namespace and it's subspaces
100*5fc512fbSwikidesign    $items = array();
101*5fc512fbSwikidesign    search($items, $dir, 'search_allpages', '');
102*5fc512fbSwikidesign
103*5fc512fbSwikidesign    // add pages with comments to result
104*5fc512fbSwikidesign    $result = array();
105*5fc512fbSwikidesign    foreach ($items as $item){
106*5fc512fbSwikidesign      $id   = ($ns ? $ns.':' : '').$item['id'];
107*5fc512fbSwikidesign
108*5fc512fbSwikidesign      // some checks
109*5fc512fbSwikidesign      $perm = auth_quickaclcheck($id);
110*5fc512fbSwikidesign      if ($perm < AUTH_READ) continue;    // skip if no permission
111*5fc512fbSwikidesign      $file = metaFN($id, '.comments');
112*5fc512fbSwikidesign      if (!@file_exists($file)) continue; // skip if no comments file
113*5fc512fbSwikidesign      $data = unserialize(io_readFile($file, false));
114*5fc512fbSwikidesign      $num = $data['number']; // skip if comments are off or closed without comments
115*5fc512fbSwikidesign      if ((!$data['status']) || (($data['status'] == 2) && (!$num))) continue;
116*5fc512fbSwikidesign
117*5fc512fbSwikidesign      $date = filemtime($file);
118*5fc512fbSwikidesign      $meta = p_get_metadata($id);
119*5fc512fbSwikidesign      $result[$date] = array(
120*5fc512fbSwikidesign        'id'       => $id,
121*5fc512fbSwikidesign        'title'    => $meta['title'],
122*5fc512fbSwikidesign        'date'     => $date,
123*5fc512fbSwikidesign        'user'     => $meta['creator'],
124*5fc512fbSwikidesign        'desc'     => $meta['description']['abstract'],
125*5fc512fbSwikidesign        'num'      => $num,
126*5fc512fbSwikidesign        'comments' => $this->td($id, $num),
127*5fc512fbSwikidesign        'perm'     => $perm,
128*5fc512fbSwikidesign        'exists'   => true,
129*5fc512fbSwikidesign      );
130*5fc512fbSwikidesign    }
131*5fc512fbSwikidesign
132*5fc512fbSwikidesign    // finally sort by time of last comment
133*5fc512fbSwikidesign    krsort($result);
134*5fc512fbSwikidesign
135*5fc512fbSwikidesign    if (is_numeric($num)) $result = array_slice($result, 0, $num);
136*5fc512fbSwikidesign
137*5fc512fbSwikidesign    return $result;
138*5fc512fbSwikidesign  }
139*5fc512fbSwikidesign
140*5fc512fbSwikidesign  /**
141*5fc512fbSwikidesign   * Returns an array of recently added comments to a given page or namespace
142*5fc512fbSwikidesign   */
143*5fc512fbSwikidesign  function getComments($ns, $num = NULL){
144*5fc512fbSwikidesign    global $conf;
145*5fc512fbSwikidesign
146*5fc512fbSwikidesign    $first  = $_REQUEST['first'];
147*5fc512fbSwikidesign    if (!is_numeric($first)) $first = 0;
148*5fc512fbSwikidesign
149*5fc512fbSwikidesign    if ((!$num) || (!is_numeric($num))) $num = $conf['recent'];
150*5fc512fbSwikidesign
151*5fc512fbSwikidesign    $result = array();
152*5fc512fbSwikidesign    $count  = 0;
153*5fc512fbSwikidesign
154*5fc512fbSwikidesign    if (!@file_exists($conf['metadir'].'/_comments.changes')) return $result;
155*5fc512fbSwikidesign
156*5fc512fbSwikidesign    // read all recent changes. (kept short)
157*5fc512fbSwikidesign    $lines = file($conf['metadir'].'/_comments.changes');
158*5fc512fbSwikidesign
159*5fc512fbSwikidesign    // handle lines
160*5fc512fbSwikidesign    for ($i = count($lines)-1; $i >= 0; $i--){
161*5fc512fbSwikidesign      $rec = _handleRecentComment($lines[$i], $ns);
162*5fc512fbSwikidesign      if ($rec !== false) {
163*5fc512fbSwikidesign        if (--$first >= 0) continue; // skip first entries
164*5fc512fbSwikidesign        $result[$rec['date']] = $rec;
165*5fc512fbSwikidesign        $count++;
166*5fc512fbSwikidesign        // break when we have enough entries
167*5fc512fbSwikidesign        if ($count >= $num) break;
168*5fc512fbSwikidesign      }
169*5fc512fbSwikidesign    }
170*5fc512fbSwikidesign
171*5fc512fbSwikidesign    // finally sort by time of last comment
172*5fc512fbSwikidesign    krsort($result);
173*5fc512fbSwikidesign
174*5fc512fbSwikidesign    return $result;
175*5fc512fbSwikidesign  }
176*5fc512fbSwikidesign
177*5fc512fbSwikidesign/* ---------- Changelog function adapted for the Discussion Plugin ---------- */
178*5fc512fbSwikidesign
179*5fc512fbSwikidesign  /**
180*5fc512fbSwikidesign   * Internal function used by $this->getComments()
181*5fc512fbSwikidesign   *
182*5fc512fbSwikidesign   * don't call directly
183*5fc512fbSwikidesign   *
184*5fc512fbSwikidesign   * @see getRecentComments()
185*5fc512fbSwikidesign   * @author Andreas Gohr <andi@splitbrain.org>
186*5fc512fbSwikidesign   * @author Ben Coburn <btcoburn@silicodon.net>
187*5fc512fbSwikidesign   * @author Esther Brunner <wikidesign@gmail.com>
188*5fc512fbSwikidesign   */
189*5fc512fbSwikidesign  function _handleRecentComment($line, $ns){
190*5fc512fbSwikidesign    static $seen  = array();         //caches seen pages and skip them
191*5fc512fbSwikidesign    if (empty($line)) return false;  //skip empty lines
192*5fc512fbSwikidesign
193*5fc512fbSwikidesign    // split the line into parts
194*5fc512fbSwikidesign    $recent = parseChangelogLine($line);
195*5fc512fbSwikidesign    if ($recent === false) return false;
196*5fc512fbSwikidesign
197*5fc512fbSwikidesign    $cid     = $recent['extra'];
198*5fc512fbSwikidesign    $fullcid = $recent['id'].'#'.$recent['extra'];
199*5fc512fbSwikidesign
200*5fc512fbSwikidesign    // skip seen ones
201*5fc512fbSwikidesign    if (isset($seen[$fullcid])) return false;
202*5fc512fbSwikidesign
203*5fc512fbSwikidesign    // skip 'show comment' log entries
204*5fc512fbSwikidesign    if ($recent['type'] === 'sc') return false;
205*5fc512fbSwikidesign
206*5fc512fbSwikidesign    // remember in seen to skip additional sights
207*5fc512fbSwikidesign    $seen[$fullcid] = 1;
208*5fc512fbSwikidesign
209*5fc512fbSwikidesign    // check if it's a hidden page or comment
210*5fc512fbSwikidesign    if (isHiddenPage($recent['id'])) return false;
211*5fc512fbSwikidesign    if ($recent['type'] === 'hc') return false;
212*5fc512fbSwikidesign
213*5fc512fbSwikidesign    // filter namespace or id
214*5fc512fbSwikidesign    if (($ns) && (strpos($recent['id'].':', $ns.':') !== 0)) return false;
215*5fc512fbSwikidesign
216*5fc512fbSwikidesign    // check ACL
217*5fc512fbSwikidesign    $recent['perm'] = auth_quickaclcheck($recent['id']);
218*5fc512fbSwikidesign    if ($recent['perm'] < AUTH_READ) return false;
219*5fc512fbSwikidesign
220*5fc512fbSwikidesign    // check existance
221*5fc512fbSwikidesign    $recent['file'] = wikiFN($recent['id']);
222*5fc512fbSwikidesign    $recent['exists'] = @file_exists($recent['file']);
223*5fc512fbSwikidesign    if (!$recent['exists']) return false;
224*5fc512fbSwikidesign    if ($recent['type'] === 'dc') return false;
225*5fc512fbSwikidesign
226*5fc512fbSwikidesign    // get discussion meta file name
227*5fc512fbSwikidesign    $data = unserialize(io_readFile(metaFN($ID, '.comments'), false));
228*5fc512fbSwikidesign
229*5fc512fbSwikidesign    // check if discussion is turned off
230*5fc512fbSwikidesign    if ($data['status'] === 0) return false;
231*5fc512fbSwikidesign
232*5fc512fbSwikidesign    // okay, then add some additional info
233*5fc512fbSwikidesign    $recent['name'] = $data['comments'][$cid]['name'];
234*5fc512fbSwikidesign    $recent['desc'] = strip_tags($data['comments'][$cid]['xhtml']);
235*5fc512fbSwikidesign
236*5fc512fbSwikidesign    return $recent;
237*5fc512fbSwikidesign  }
238*5fc512fbSwikidesign
239*5fc512fbSwikidesign}
240*5fc512fbSwikidesign
241*5fc512fbSwikidesign//Setup VIM: ex: et ts=4 enc=utf-8 :
242