1<?php
2/**
3 * Plugin catlist : Displays a list of the pages of a namespace recursively
4 *
5 * @license   MIT
6 * @author    FĂ©lix Faisant <xcodexif@xif.fr>
7 *
8 */
9
10use dokuwiki\File\PageResolver;
11use dokuwiki\Utf8\PhpString;
12
13if (!defined('DOKU_INC')) die('meh.');
14
15define('CATLIST_DISPLAY_LIST', 1);
16define('CATLIST_DISPLAY_LINE', 2);
17
18define('CATLIST_NSLINK_AUTO', 0);
19define('CATLIST_NSLINK_NONE', 1);
20define('CATLIST_NSLINK_FORCE', 2);
21
22define('CATLIST_INDEX_START', 0);
23define('CATLIST_INDEX_OUTSIDE', 1);
24define('CATLIST_INDEX_INSIDE', 2);
25
26define('CATLIST_SORT_NONE', 0);
27define('CATLIST_SORT_ASCENDING', 1);
28define('CATLIST_SORT_DESCENDING', 2);
29
30class syntax_plugin_catlist extends DokuWiki_Syntax_Plugin {
31
32	function connectTo ($aMode) {
33		$this->Lexer->addSpecialPattern('<catlist[^>]*>', $aMode, 'plugin_catlist');
34	}
35
36	function getSort () {
37		return 189;
38	}
39
40	function getType () {
41		return 'substition';
42	}
43
44	/*********************************************************************************************/
45	/************************************ <catlist> directive ************************************/
46
47	function _checkOption(&$match, $option, &$varAffected, $valIfFound){
48		if (preg_match('/-'.$option.' /i', $match, $found)) {
49			$varAffected = $valIfFound;
50			$match = str_replace($found[0], '', $match);
51		}
52	}
53	function _checkOptionParam(&$match, $option, &$varAffected, $varAssoc){
54		if (preg_match('/-'.$option.':('.implode('|',array_keys($varAssoc)).') /i', $match, $found)) {
55			$varAffected = $varAssoc[$found[1]];
56			$match = str_replace($found[0], '', $match);
57		}
58	}
59
60	function handle ($match, $state, $pos, Doku_Handler $handler) {
61		global $conf;
62
63		$_default_sort_map = array("none" => CATLIST_SORT_NONE,
64		                           "ascending" => CATLIST_SORT_ASCENDING,
65		                           "descending" => CATLIST_SORT_DESCENDING);
66		$_index_priority_map = array("start" => CATLIST_INDEX_START,
67		                             "outside" => CATLIST_INDEX_OUTSIDE,
68		                             "inside" => CATLIST_INDEX_INSIDE);
69
70		$data = array('displayType' => CATLIST_DISPLAY_LIST, 'nsInBold' => true, 'expand' => 6,
71		              'exclupage' => array(), 'excluns' => array(), 'exclunsall' => array(), 'exclunspages' => array(), 'exclunsns' => array(),
72		              'exclutype' => 'id',
73		              'createPageButtonNs' => true, 'createPageButtonSubs' => false, 'pagename_sanitize' => (boolean)$this->getConf('pagename_sanitize'),
74		              'head' => (boolean)$this->getConf('showhead'),
75		              'headTitle' => NULL, 'smallHead' => false, 'linkStartHead' => true, 'hn' => 'h1',
76		              'useheading' => (boolean)$this->getConf('useheading'),
77		              'nsuseheading' => NULL, 'nsLinks' => CATLIST_NSLINK_AUTO,
78		              'columns' => 0, 'maxdepth' => 0,
79		              'sort_order' => $_default_sort_map[$this->getConf('default_sort')],
80		              'sort_by_title' => false, 'sort_by_type' => false, 'sort_by_date' => false, 'sort_collator' => $this->getConf('sort_collator_locale'),
81		              'hide_index' => (boolean)$this->getConf('hide_index'),
82		              'index_priority' => array(),
83		              'nocache' => (boolean)$this->getConf('nocache'),
84		              'hide_nsnotr' => (boolean)$this->getConf('hide_acl_nsnotr'), 'show_pgnoread' => false, 'show_perms' => (boolean)$this->getConf('show_acl'),
85		              'show_leading_ns' => (boolean)$this->getConf('show_leading_ns'),
86		              'show_notfound_error' => true );
87
88		$index_priority = explode(',', $this->getConf('index_priority'));
89		foreach ($index_priority as $index_type) {
90			if (!array_key_exists($index_type, $_index_priority_map)) {
91				msg("catlist: invalid index type in index_priority", -1);
92				return false;
93			}
94			$data['index_priority'][] = $_index_priority_map[$index_type];
95		}
96		$match = PhpString::substr($match, 9, -1) . ' ';
97
98		// Display options
99		$this->_checkOption($match, "displayList", $data['displayType'], CATLIST_DISPLAY_LIST);
100		$this->_checkOption($match, "displayLine", $data['displayType'], CATLIST_DISPLAY_LINE);
101		$this->_checkOption($match, "noNSInBold", $data['nsInBold'], false);
102		if (preg_match("/-expandButton:([0-9]+)/i", $match, $found)) {
103			$data['expand'] = intval($found[1]);
104			$match = str_replace($found[0], '', $match);
105		}
106		$this->_checkOption($match, "noHeadTitle", $data['useheading'], false);
107		$this->_checkOption($match, "forceHeadTitle", $data['useheading'], true);
108		$data['nsuseheading'] = $data['useheading'];
109		$this->_checkOption($match, "noNSHeadTitle", $data['nsuseheading'], false);
110		$this->_checkOption($match, "hideNotFoundMsg", $data['show_notfound_error'], false);
111
112		// Namespace options
113		$this->_checkOption($match, "forceLinks", $data['nsLinks'], CATLIST_NSLINK_FORCE); // /!\ Deprecated
114		$this->_checkOptionParam($match, "nsLinks", $data['nsLinks'], array( "none" => CATLIST_NSLINK_NONE,
115		                                                                     "auto" => CATLIST_NSLINK_AUTO,
116		                                                                     "force" => CATLIST_NSLINK_FORCE ));
117
118		// Exclude options
119		for ($found; preg_match("/-(exclu(page|ns|nsall|nspages|nsns)!?):\"([^\\/\"]+)\" /i", $match, $found); ) {
120			$option = strtolower($found[1]);
121			// is regex negated ?
122			if (substr($option,-1) == "!") {
123				$data[substr($option,0,-1)][] = array('regex' => $found[3], 'neg' => true);
124			} else {
125				$data[$option][] = array('regex' => $found[3], 'neg' => false);
126			}
127			$match = str_replace($found[0], '', $match);
128		}
129		for ($found; preg_match("/-(exclu(page|ns|nsall|nspages|nsns)) /i", $match, $found); ) {
130			$data[strtolower($found[1])] = true;
131			$match = str_replace($found[0], '', $match);
132		}
133		// Exclude type (exclude based on id, name, or title)
134		$this->_checkOption($match, "excludeOnID", $data['exclutype'], 'id');
135		$this->_checkOption($match, "excludeOnName", $data['exclutype'], 'name');
136		$this->_checkOption($match, "excludeOnTitle", $data['exclutype'], 'title');
137		// Exclude page/namespace id list
138		$data['excludelist'] = array();
139		for ($found; preg_match("/-exclude:\\{([^\\}]*)\\} /", $match, $found); ) {
140			$list = explode(' ', $found[1]);
141			$data['excludelist'] = array_merge($data['excludelist'], $list);
142			$match = str_replace($found[0], '', $match);
143		}
144
145		// Max depth
146		if (preg_match("/-maxDepth:([0-9]+)/i", $match, $found)) {
147			$data['maxdepth'] = intval($found[1]);
148			$match = str_replace($found[0], '', $match);
149		}
150
151		// Columns
152		if (preg_match("/-columns:([0-9]+)/i", $match, $found)) {
153			$data['columns'] = intval($found[1]);
154			$match = str_replace($found[0], '', $match);
155		}
156
157		// Head options
158		$this->_checkOption($match, "noHead", $data['head'], false);
159		$this->_checkOption($match, "showHead", $data['head'], true);
160		$this->_checkOption($match, "smallHead", $data['smallHead'], true);
161		$this->_checkOption($match, "noLinkStartHead", $data['linkStartHead'], false);
162		if (preg_match("/-(h[1-5])/i", $match, $found)) {
163			$data['hn'] = $found[1];
164			$match = str_replace($found[0], '', $match);
165		}
166		if (preg_match("/-titleHead:\"([^\"]*)\"/i", $match, $found)) {
167			$data['headTitle'] = $found[1];
168			$match = str_replace($found[0], '', $match);
169		}
170
171		// Create page button options
172		$this->_checkOption($match, "noAddPageButton", $data['createPageButtonNs'], false);
173		$this->_checkOption($match, "addPageButtonEach", $data['createPageButtonSubs'], true);
174
175		// Sorting options
176		$this->_checkOption($match, "sortAscending", $data['sort_order'], CATLIST_SORT_ASCENDING);
177		$this->_checkOption($match, "sortDescending", $data['sort_order'], CATLIST_SORT_DESCENDING);
178		$this->_checkOption($match, "sortByTitle", $data['sort_by_title'], true);
179		$this->_checkOption($match, "sortByType", $data['sort_by_type'], true);
180		$this->_checkOption($match, "sortByCreationDate", $data['sort_by_date'], 'created');
181		$this->_checkOption($match, "sortByModifDate", $data['sort_by_date'], 'modified');
182
183		// ACL options
184		$this->_checkOption($match, "ACLshowPage", $data['show_pgnoread'], true);
185		$this->_checkOption($match, "ACLhideNs", $data['hide_nsnotr'], true);
186
187		// Remove other options and warn about
188		for ($found; preg_match("/ (-.*)/", $match, $found); ) {
189			msg(sprintf($this->getLang('unknownoption'), htmlspecialchars($found[1])), -1);
190			$match = str_replace($found[0], '', $match);
191		}
192
193		// Looking for the wanted namespace. Now, only the wanted namespace remains in $match. Then clean the namespace id
194		$ns = trim($match);
195		if ((boolean)$this->getConf('nswildcards')) {
196			global $ID;
197			$parsepagetemplate_data = array('id' => $ID, 'tpl' => $ns, 'doreplace' => true);
198			$ns = parsePageTemplate($parsepagetemplate_data);
199		}
200		if ($ns == '') $ns = '.'; // If there is nothing, we take the current namespace
201		global $ID;
202		if ($ns[0] == '.') $ns = getNS($ID).':'.$ns; // If it start with a '.', it is a relative path
203		$split = explode(':', $ns);
204		for ($i = 0; $i < count($split); $i++) {
205			if ($split[$i] === '' || $split[$i] === '.') {
206				array_splice($split, $i, 1);
207				$i--;
208			} else if ($split[$i] == '..') {
209				if ($i != 0) {
210					array_splice($split, $i-1, 2);
211					$i -= 2;
212				} else break;
213			}
214		}
215		if (count($split) > 0 && $split[0] == '..') {
216			// Path would be outside the 'pages' directory
217			msg($this->getLang('outofpages'), -1);
218			return false;
219		}
220		$data['ns'] = implode(':', $split);
221		return $data;
222	}
223
224	/**************************************************************************************/
225	/************************************ Tree walking ************************************/
226
227		/* Utility function to check is a given page/namespace ($item) is excluded
228		 * based on the relevant list of blacklisting/whitelisting regexes $arrayRegex
229		 * ( array of array('regex'=>the_regex,'neg'=>false/true) ). The exclusion
230		 * is based on item title, full id or name ($exclutype).
231		 */
232	function _isExcluded ($item, $exclutype, $arrayRegex) {
233		if ($arrayRegex === true) return true;
234		global $conf;
235		if ((strlen($conf['hidepages']) != 0) && preg_match('/'.$conf['hidepages'].'/i', $item['id'])) return true;
236		foreach($arrayRegex as $regex) {
237			if (!is_array($regex)) // temporary, for transitioning to v2021-07-21
238				$regex = array('regex' => $regex, 'neg' => false);
239			$match = preg_match('/'.$regex['regex'].(($exclutype=='title')?'/':'/i'), $item[$exclutype]);
240			if ($regex['neg']) {
241				if ($match === 0)
242					return true;
243			} else {
244				if ($match === 1)
245					return true;
246			}
247		}
248		return false;
249	}
250
251	function _getStartPage ($index_priority, $parid, $parpath, $name, $force) {
252		$exists = false;
253		if ($parid != '') $parid .= ':';
254		global $conf;
255		$index_path_map = array( CATLIST_INDEX_START => $parpath.'/'.$name.'/'.$conf['start'].'.txt',
256		                         CATLIST_INDEX_OUTSIDE => $parpath.'/'.$name.'.txt',
257		                         CATLIST_INDEX_INSIDE => $parpath.'/'.$name.'/'.$name.'.txt' );
258		$index_id_map = array( CATLIST_INDEX_START => $parid .$name.':'.$conf['start'],
259		                       CATLIST_INDEX_OUTSIDE => $parid .$name,
260		                       CATLIST_INDEX_INSIDE => $parid .$name.':'.$name );
261		foreach ($index_priority as $index_type) {
262			if (is_file($index_path_map[$index_type])) {
263				$exists = true;
264				return array(true, $index_id_map[$index_type], $index_path_map[$index_type]);
265			}
266		}
267		if ($force && isset($index_priority[0]))
268			return array(false, $index_id_map[0], null);
269		else
270			return array(false, false, null);
271		// returns ($index_exists, $index_id, $index_filepath)
272	}
273
274	function _getMetadata ($id, $filepath) {
275		$meta = p_get_metadata($id, $key='', $render=METADATA_RENDER_USING_SIMPLE_CACHE);
276		if (!isset($meta['date']['modified']))
277			$meta['date']['modified'] = @filemtime($filepath);
278		if (!isset($meta['contributor']))
279			$meta['contributor'] = $meta['creator'];
280		return $meta;
281	}
282
283		/* Entry function for tree walking, called in render()
284		 *
285		 * $data contains the various options initialized and parsed in handle(), and will be passed along
286		 * the tree walking. Moreover, $data['tree'] is filled by the pages found by _walk_recurse(), and
287		 * will contain the full tree, minus the excluded pages (however, permissions are only evaluated at
288		 * rendering time) and up to the max depth. _walk() prepares and start the tree walking.
289		 */
290	function _walk (&$data) {
291		global $conf;
292
293			// Get the directory path from namespace id, and check if it exists
294		$ns = $data['ns'];
295		$path = str_replace(':', '/', $ns);
296		$path = $conf['datadir'].'/'.utf8_encodeFN($path);
297		if (!is_dir($path)) {
298			if ($data['show_notfound_error'])
299				msg(sprintf($this->getLang('dontexist'), $ns), -1);
300			return false;
301		}
302
303			// Info on the main page (the "header" page)
304		$id = $ns . ':';
305		$resolver = new PageResolver($id);
306		$id = $resolver->resolveId($id);
307		$main = array( 'id' => $id,
308		               'exist' => page_exists($id),
309		               'title' => NULL );
310		if ($data['headTitle'] !== NULL)
311			$main['title'] = $data['headTitle'];
312		else {
313			if ($data['useheading'] && $main['exist'])
314				$main['title'] = p_get_first_heading($main['id'], true);
315			if (is_null($main['title'])) {
316				$ex = explode(':', $ns);
317				$main['title'] = end($ex);
318			}
319		}
320		$data['main'] = $main;
321
322			// Preparing other stuff
323		if (!isset($data['sort_collator']) || $data['sort_collator'] == "")
324			$data['sort_collator'] = NULL;
325		else {
326			$locale = $data['sort_collator'];
327			$coll = collator_create($locale);
328			if (!isset($coll)) {
329				msg("catlist sorting: can't create Collator object: ".intl_get_error_message(), -1);
330				$data['sort_collator'] = NULL;
331			} else {
332				$coll->setAttribute(Collator::CASE_FIRST, Collator::UPPER_FIRST);
333				$coll->setAttribute(Collator::NUMERIC_COLLATION, Collator::ON);
334				$data['sort_collator'] = $coll;
335			}
336		}
337
338			// Start the recursion
339		if (!isset($data['excludelist'])) // temporary, for transitioning to v2021-07-21
340			$data['excludelist'] = array();
341		$data['tree'] = array();
342		$data['index_pages'] = array( $main['id'] );
343		$this->_walk_recurse($data, $path, $ns, "", false, false, 1/*root depth is 1*/, $data['tree']/*root*/);
344		return true;
345	}
346
347		/* Recursive function for tree walking.
348		 *
349		 * Scans the current namespace by looking directly at the filesystem directory
350		 * for .txt files (pages) and sub-directories (namespaces). Excludes pages/namespaces
351		 * based on the various exclusion options. The current/local directory path, namesapce
352		 * ID and relative namespace ID are respectively $path, $ns and $relns.
353		 * $data is described above. $data['tree'] is not modified directly, but only through
354		 * $_TREE which is the *local* tree view (ie. a reference of a $data['tree'] node) and
355		 * where found children are added. Optionally sorts this list of children.
356		 * The local tree depth is $depth. $excluPages, $excluNS are flags indicates if the
357		 * sub-pages/namespaces should be excluded. Fills $data['index_pages'] with all
358		 * namespace IDs where an index has been found.
359		 */
360	function _walk_recurse (&$data, $path, $ns, $relns, $excluPages, $excluNS, $depth, &$_TREE) {
361		$scanDirs = @scandir($path, SCANDIR_SORT_NONE);
362		if ($scanDirs === false) {
363			msg("catlist: can't open directory of namespace ".$ns, -1);
364			return;
365		}
366		foreach ($scanDirs as $file) {
367			if ($file[0] == '.' || $file[0] == '_') continue;
368			$name = utf8_decodeFN(str_replace('.txt', '', $file));
369			$id = ($ns == '') ? $name : $ns.':'.$name;
370			$rel_id = ($relns == '') ? $name : $relns.':'.$name;
371			$item = array('id' => $id, 'rel_id' => $rel_id, 'name' => $name, 'title' => NULL);
372
373				// ID exclusion
374			if (in_array($rel_id, $data['excludelist'])) continue;
375
376				// It's a namespace
377			if (is_dir($path.'/'.$file)) {
378					// Index page of the namespace
379				list($index_exists, $index_id, $index_filepath) = $this->_getStartPage($data['index_priority'], $ns, $path, $name, ($data['nsLinks']==CATLIST_NSLINK_FORCE));
380				if ($index_exists)
381					$data['index_pages'][] = $index_id;
382					// Exclusion
383				if ($excluNS) continue;
384				if ($this->_isExcluded($item, $data['exclutype'], $data['excluns'])) continue;
385					// Namespace
386				if ($index_exists) {
387					$item['metadata'] = $this->_getMetadata($index_id, $index_filepath);
388					if ($data['nsuseheading'] && isset($item['metadata']['title']))
389						$item['title'] = $item['metadata']['title'];
390				}
391				if (is_null($item['title']))
392					$item['title'] = $name;
393				$item['linkdisp'] = ($index_exists && ($data['nsLinks']==CATLIST_NSLINK_AUTO)) || ($data['nsLinks']==CATLIST_NSLINK_FORCE);
394				$item['linkid'] = $index_id;
395					// Button
396				$item['buttonid'] = $data['createPageButtonSubs'] ? $id.':' : NULL;
397					// Recursion if wanted
398				$item['_'] = array();
399				$okdepth = ($depth < $data['maxdepth']) || ($data['maxdepth'] == 0);
400				$exclude_content = $this->_isExcluded($item, $data['exclutype'], $data['exclunsall'])
401				                   || in_array($rel_id.':', $data['excludelist']);
402				if (!$exclude_content && $okdepth) {
403					$exclunspages = $this->_isExcluded($item, $data['exclutype'], $data['exclunspages']);
404					$exclunsns = $this->_isExcluded($item, $data['exclutype'], $data['exclunsns']);
405					$this->_walk_recurse($data, $path.'/'.$file, $id, $rel_id, $exclunspages, $exclunsns, $depth+1, $item['_']);
406				}
407					// Tree
408				$_TREE[] = $item;
409			} else
410
411				// It's a page
412			if (!$excluPages) {
413				if (substr($file, -4) != ".txt") continue;
414					// Page title
415				$item['metadata'] = $this->_getMetadata($id, $file);
416				if ($data['useheading'] && isset($item['metadata']['title'])) {
417					$item['title'] = $item['metadata']['title'];
418				}
419				if (is_null($item['title']))
420					$item['title'] = $name;
421					// Exclusion
422				if ($this->_isExcluded($item, $data['exclutype'], $data['exclupage'])) continue;
423					// Tree
424				$_TREE[] = $item;
425			}
426
427				// Sorting
428			if ($data['sort_order'] != CATLIST_SORT_NONE) {
429				usort($_TREE, function ($a, $b) use ($data) {
430					$a_is_folder = isset($a['_']);
431					$b_is_folder = isset($b['_']);
432					// if one or the other is folder, comparison is done
433					if ($data['sort_by_type'] && ($a_is_folder xor $b_is_folder ))
434						return $b_is_folder;
435					// else, compare date or name
436					if ($data['sort_by_date'] === false) {
437						// by name
438						$a_title = ($data['sort_by_title'] ? $a['title'] : $a['name']);
439						$b_title = ($data['sort_by_title'] ? $b['title'] : $b['name']);
440						if (!is_null($data['sort_collator']))
441							$r = $data['sort_collator']->compare($a_title, $b_title);
442						else
443							$r = strnatcasecmp($a_title, $b_title);
444					} else {
445						// by date
446						$field = $data['sort_by_date'];
447						$a_date = (isset($a['metadata']['date'][$field]) ? $a['metadata']['date'][$field] : 0);
448						$b_date = (isset($b['metadata']['date'][$field]) ? $b['metadata']['date'][$field] : 0);
449						$r = $a_date <=> $b_date;
450					}
451					if ($data['sort_order'] == CATLIST_SORT_DESCENDING)
452						$r *= -1;
453					return $r;
454				});
455			}
456		}
457	}
458
459	/***********************************************************************************/
460	/************************************ Rendering ************************************/
461
462	function render ($mode, Doku_Renderer $renderer, $data) {
463		if (!is_array($data)) return false;
464
465		if ($data['ns'] == '%%CURRENT_NAMESPACE%%')
466			$data['ns'] = getNS(cleanID(getID())); // update namespace to the one currently displayed
467		$ns = $data['ns'];
468
469			// Disabling cache
470		if ($data['nocache'])
471			$renderer->nocache();
472
473			// Walk namespace tree
474		$r = $this->_walk($data);
475		if ($r == false) return false;
476
477			// Write params for the add page button
478		global $conf;
479		if (!isset($data['pagename_sanitize'])) // temporary, for transitioning to v2022-06-25
480			$data['pagename_sanitize'] = true;
481		$renderer->doc .= '<script type="text/javascript"> catlist_baseurl = "'.DOKU_URL.'"; catlist_basescript = "'.DOKU_SCRIPT.'"; catlist_useslash = '.$conf['useslash'].'; catlist_userewrite = '.$conf['userewrite'].'; catlist_sepchar = "'.$conf['sepchar'].'"; catlist_deaccent = '.$conf['deaccent'].'; catlist_pagename_sanitize = '.$data['pagename_sanitize'].'; </script>';
482
483			// Display headline
484		if ($data['head']) {
485			$html_tag_small = ($data['nsInBold']) ? 'strong' : 'span';
486			$html_tag = ($data['smallHead']) ? $html_tag_small : $data['hn'];
487			$renderer->doc .= '<'.$html_tag.' class="catlist-head">';
488			$main = $data['main'];
489			if (($main['exist'] && $data['linkStartHead'] && !($data['nsLinks']==CATLIST_NSLINK_NONE)) || ($data['nsLinks']==CATLIST_NSLINK_FORCE))
490				$renderer->internallink(':'.$main['id'], $main['title']);
491			else
492				$renderer->doc .= htmlspecialchars($main['title']);
493			$renderer->doc .= '</'.$html_tag.'>';
494		}
495
496			// Recurse and display
497		$global_ul_attr = "";
498		if ($data['columns'] != 0) {
499			$global_ul_attr = 'column-count: '.$data['columns'].';';
500			$global_ul_attr = 'style="-webkit-'.$global_ul_attr.' -moz-'.$global_ul_attr.' '.$global_ul_attr.'" ';
501			$global_ul_attr .= 'class="catlist_columns catlist-nslist" ';
502		} else {
503			$global_ul_attr = 'class="catlist-nslist" ';
504		}
505		if ($data['displayType'] == CATLIST_DISPLAY_LIST) $renderer->doc .= '<ul '.$global_ul_attr.'>';
506		$this->_recurse($renderer, $data, $data['tree']);
507		$perm_create = $this->_cached_quickaclcheck($ns.':*') >= AUTH_CREATE;
508		$ns_button = ($ns == '') ? '' : $ns.':';
509		if ($data['createPageButtonNs'] && $perm_create)
510			$this->_displayAddPageButton($renderer, $ns_button, $data['displayType']);
511		if ($data['displayType'] == CATLIST_DISPLAY_LIST)
512			$renderer->doc .= '</ul>';
513
514		return true;
515	}
516
517		/* Just cache the calls to auth_quickaclcheck, mainly for _any_child_perms */
518	function _cached_quickaclcheck($id) {
519		static $cache = array();
520		if (!isset($cache[$id]))
521			$cache[$id] = auth_quickaclcheck($id);
522		return $cache[$id];
523	}
524
525		/* Walk the tree to see if any page/namespace below this has read access access, for show_leading_ns option */
526	function _any_child_perms ($data, $_TREE) {
527		foreach ($_TREE as $item) {
528			if (isset($item['_'])) {
529				$perms = $this->_cached_quickaclcheck($item['id'].':*');
530				if ($perms >= AUTH_READ || $this->_any_child_perms($data, $item['_']))
531					return true;
532			} else {
533				$perms = $this->_cached_quickaclcheck($item['id']);
534				if ($perms >= AUTH_READ)
535					return true;
536			}
537		}
538		return false;
539	}
540
541	function _recurse (&$renderer, $data, $_TREE) {
542		foreach ($_TREE as $item) {
543			if (isset($item['_'])) {
544				// It's a namespace
545				$perms = $this->_cached_quickaclcheck($item['id'].':*');
546				$perms_exemption = $data['show_perms'];
547				// If we actually care about not showing the namespace because of permissions :
548				if ($perms < AUTH_READ && !$perms_exemption) {
549					// If show_leading_ns activated, walk the tree below this, see if any page/namespace below this has access
550					if ($data['show_leading_ns'] && $this->_any_child_perms($data, $item['_'])) {
551						$perms_exemption = true;
552					} else {
553						if ($data['hide_nsnotr']) continue;
554						if ($data['show_pgnoread'])
555							$perms_exemption = true; // Add exception if show_pgnoread enabled, but hide_nsnotr prevails
556					}
557				}
558				$linkdisp = $item['linkdisp'] && ($perms >= AUTH_READ);
559				if ($perms < AUTH_CREATE)
560					$item['buttonid'] = NULL;
561				$this->_displayNSBegin($renderer, $data, $item['title'], $linkdisp, $item['linkid'], ($data['show_perms'] ? $perms : NULL));
562				if ($perms >= AUTH_READ || $perms_exemption)
563					$this->_recurse($renderer, $data, $item['_']);
564				$this->_displayNSEnd($renderer, $data['displayType'], $item['buttonid']);
565			} else {
566				// It's a page
567				$perms = $this->_cached_quickaclcheck($item['id']);
568				if ($perms < AUTH_READ && !$data['show_perms'] && !$data['show_pgnoread'])
569					continue;
570				if ($data['hide_index'] && in_array($item['id'], $data['index_pages']))
571					continue;
572				$displayLink = $perms >= AUTH_READ || $data['show_perms'];
573				$this->_displayPage($renderer, $item, $data['displayType'], ($data['show_perms'] ? $perms : NULL), $displayLink);
574			}
575		}
576	}
577
578	function _displayNSBegin (&$renderer, $data, $title, $displayLink, $idLink, $perms) {
579		if ($data['displayType'] == CATLIST_DISPLAY_LIST) {
580			$warper_ns = ($data['nsInBold']) ? 'strong' : 'span';
581			$renderer->doc .= '<li class="catlist-ns"><'.$warper_ns.' class="li catlist-nshead">';
582			if ($displayLink) $renderer->internallink($idLink, $title);
583			else $renderer->doc .= htmlspecialchars($title);
584			if ($perms !== NULL) $renderer->doc .= ' [ns, perm='.$perms.']';
585			$renderer->doc .= '</'.$warper_ns.'>';
586			$renderer->doc .= '<ul class="catlist-nslist">';
587		}
588		else if ($data['displayType'] == CATLIST_DISPLAY_LINE) {
589			if ($data['nsInBold']) $renderer->doc .= '<strong>';
590			if ($displayLink) $renderer->internallink($idLink, $title);
591			else $renderer->doc .= htmlspecialchars($title);
592			if ($data['nsInBold']) $renderer->doc .= '</strong>';
593			$renderer->doc .= '[ ';
594		}
595	}
596
597	function _displayNSEnd (&$renderer, $displayType, $nsAddButton) {
598		if (!is_null($nsAddButton)) $this->_displayAddPageButton($renderer, $nsAddButton, $displayType);
599		if ($displayType == CATLIST_DISPLAY_LIST) $renderer->doc .= '</ul></li>';
600		else if ($displayType == CATLIST_DISPLAY_LINE) $renderer->doc .= '] ';
601	}
602
603	function _displayPage (&$renderer, $item, $displayType, $perms, $displayLink) {
604		if ($displayType == CATLIST_DISPLAY_LIST) {
605			$renderer->doc .= '<li class="catlist-page">';
606			if ($displayLink) $renderer->internallink(':'.$item['id'], $item['title']);
607			else $renderer->doc .= htmlspecialchars($item['title']);
608			if ($perms !== NULL) $renderer->doc .= ' [page, perm='.$perms.']';
609			$renderer->doc .= '</li>';
610		} else if ($displayType == CATLIST_DISPLAY_LINE) {
611			$renderer->internallink(':'.$item['id'], $item['title']);
612			$renderer->doc .= ' ';
613		}
614	}
615
616	function _displayAddPageButton (&$renderer, $ns, $displayType) {
617		$html = ($displayType == CATLIST_DISPLAY_LIST) ? 'li' : 'span';
618		$renderer->doc .= '<'.$html.' class="catlist_addpage"><button class="button" onclick="catlist_button_add_page(this,\''.$ns.'\')">'.$this->getLang('addpage').'</button></'.$html.'>';
619	}
620
621}
622