1f9f65fc3Swikidesign<?php 2f9f65fc3Swikidesign/** 3f9f65fc3Swikidesign * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 4f9f65fc3Swikidesign * @author Esther Brunner <wikidesign@gmail.com> 5f9f65fc3Swikidesign */ 6f9f65fc3Swikidesign 7f9f65fc3Swikidesign// must be run within Dokuwiki 8f9f65fc3Swikidesignif (!defined('DOKU_INC')) die(); 9f9f65fc3Swikidesign 10f9f65fc3Swikidesignif (!defined('DOKU_LF')) define('DOKU_LF', "\n"); 11f9f65fc3Swikidesignif (!defined('DOKU_TAB')) define('DOKU_TAB', "\t"); 12f9f65fc3Swikidesignif (!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN', DOKU_INC.'lib/plugins/'); 13f9f65fc3Swikidesign 14f9f65fc3Swikidesignclass helper_plugin_include extends DokuWiki_Plugin { // DokuWiki_Helper_Plugin 15f9f65fc3Swikidesign 16b6337848Swikidesign var $pages = array(); // filechain of included pages 17f9f65fc3Swikidesign var $page = array(); // associative array with data about the page to include 18f9f65fc3Swikidesign var $ins = array(); // instructions array 19f9f65fc3Swikidesign var $doc = ''; // the final output XHTML string 20f9f65fc3Swikidesign var $mode = 'section'; // inclusion mode: 'page' or 'section' 21f9f65fc3Swikidesign var $clevel = 0; // current section level 22f9f65fc3Swikidesign var $firstsec = 0; // show first section only 23f9f65fc3Swikidesign var $hasheader = 0; // included page has header 24f9f65fc3Swikidesign 25f9f65fc3Swikidesign function getInfo(){ 26f9f65fc3Swikidesign return array( 27f9f65fc3Swikidesign 'author' => 'Esther Brunner', 28f9f65fc3Swikidesign 'email' => 'wikidesign@gmail.com', 296cad866dSwikidesign 'date' => '2006-12-18', 30f9f65fc3Swikidesign 'name' => 'Include Plugin (helper class)', 31f9f65fc3Swikidesign 'desc' => 'Functions to include another page in a wiki page', 32f9f65fc3Swikidesign 'url' => 'http://www.wikidesign/en/plugin/include/start', 33f9f65fc3Swikidesign ); 34f9f65fc3Swikidesign } 35f9f65fc3Swikidesign 36f9f65fc3Swikidesign function getMethods(){ 37f9f65fc3Swikidesign $result = array(); 38f9f65fc3Swikidesign $result[] = array( 39f9f65fc3Swikidesign 'name' => 'setPage', 40f9f65fc3Swikidesign 'desc' => 'sets the page to include', 41f9f65fc3Swikidesign 'params' => array("page attributes, 'id' required, 'section' for filtering" => 'array'), 42f9f65fc3Swikidesign 'return' => array('success' => 'boolean'), 43f9f65fc3Swikidesign ); 44f9f65fc3Swikidesign $result[] = array( 45f9f65fc3Swikidesign 'name' => 'setMode', 46f9f65fc3Swikidesign 'desc' => 'sets inclusion mode: should indention be merged?', 47f9f65fc3Swikidesign 'params' => array("'page' (original) or 'section' (merged indention)" => 'string'), 48f9f65fc3Swikidesign ); 49f9f65fc3Swikidesign $result[] = array( 50f9f65fc3Swikidesign 'name' => 'setLevel', 51f9f65fc3Swikidesign 'desc' => 'sets the indention for the current section level', 52f9f65fc3Swikidesign 'params' => array('level: 0 to 5' => 'integer'), 53f9f65fc3Swikidesign 'return' => array('success' => 'boolean'), 54f9f65fc3Swikidesign ); 55f9f65fc3Swikidesign $result[] = array( 56b6337848Swikidesign 'name' => 'renderXHTML', 57b6337848Swikidesign 'desc' => 'renders the XHTML output of the included page', 58f9f65fc3Swikidesign 'params' => array('DokuWiki renderer' => 'object'), 59b6337848Swikidesign 'return' => array('XHTML' => 'string'), 60f9f65fc3Swikidesign ); 61f9f65fc3Swikidesign return $result; 62f9f65fc3Swikidesign } 63f9f65fc3Swikidesign 64f9f65fc3Swikidesign /** 65f9f65fc3Swikidesign * Sets the page to include if it is not already included (prevent recursion) 66f9f65fc3Swikidesign */ 67f9f65fc3Swikidesign function setPage($page){ 68f9f65fc3Swikidesign global $ID; 69f9f65fc3Swikidesign 70b6337848Swikidesign $id = $page['id']; 71b6337848Swikidesign $fullid = $id.'#'.$page['section']; 72f9f65fc3Swikidesign 73b6337848Swikidesign if (!$id) return false; // no page id given 74b6337848Swikidesign if ($id == $ID) return false; // page can't include itself 75f9f65fc3Swikidesign 76b6337848Swikidesign // prevent include recursion 77b6337848Swikidesign if ((isset($this->pages[$id.'#'])) || (isset($this->pages[$fullid]))) return false; 78b6337848Swikidesign 79b6337848Swikidesign // add the page to the filechain 80b6337848Swikidesign $this->pages[$fullid] = $page; 81b6337848Swikidesign $this->page =& $this->pages[$fullid]; 82f9f65fc3Swikidesign return true; 83f9f65fc3Swikidesign } 84f9f65fc3Swikidesign 85f9f65fc3Swikidesign /** 86f9f65fc3Swikidesign * Sets the inclusion mode 87f9f65fc3Swikidesign */ 88f9f65fc3Swikidesign function setMode($mode){ 89f9f65fc3Swikidesign $this->mode = $mode; 90f9f65fc3Swikidesign } 91f9f65fc3Swikidesign 92f9f65fc3Swikidesign /** 93f9f65fc3Swikidesign * Sets the right indention for a given section level 94f9f65fc3Swikidesign */ 95f9f65fc3Swikidesign function setLevel($level){ 96f9f65fc3Swikidesign if ((is_numeric($level)) && ($level >= 0) && ($level <= 5)){ 97f9f65fc3Swikidesign $this->clevel = $level; 98f9f65fc3Swikidesign return true; 99f9f65fc3Swikidesign } 100f9f65fc3Swikidesign return false; 101f9f65fc3Swikidesign } 102f9f65fc3Swikidesign 103f9f65fc3Swikidesign /** 104f9f65fc3Swikidesign * Builds the XHTML to embed the page to include 105f9f65fc3Swikidesign */ 106b6337848Swikidesign function renderXHTML(&$renderer){ 107b6337848Swikidesign if (!$this->page['id']) return ''; // page must be set first 108f9f65fc3Swikidesign 109f9f65fc3Swikidesign $this->doc = ''; 110f9f65fc3Swikidesign $this->firstsec = $this->getConf('firstseconly'); 111f9f65fc3Swikidesign 112f9f65fc3Swikidesign // get instructions and render them on the fly 113f9f65fc3Swikidesign $this->page['file'] = wikiFN($this->page['id']); 114f9f65fc3Swikidesign $this->ins = p_cached_instructions($this->page['file']); 115f9f65fc3Swikidesign 116f9f65fc3Swikidesign // show only a given section? 117f9f65fc3Swikidesign if ($this->page['section']) $this->_getSection(); 118f9f65fc3Swikidesign 119f9f65fc3Swikidesign // convert relative links 120f9f65fc3Swikidesign $this->_convertInstructions($renderer); 121f9f65fc3Swikidesign 122f9f65fc3Swikidesign // insert a read more link if only first section is shown 123f9f65fc3Swikidesign if ($this->firstsec) $this->_readMore(); 124f9f65fc3Swikidesign 125f9f65fc3Swikidesign // render the included page 126f9f65fc3Swikidesign $content = $this->_cleanXHTML(p_render('xhtml', $this->ins, $info)); 127f9f65fc3Swikidesign 128f9f65fc3Swikidesign // embed the included page 129b6337848Swikidesign $renderer->doc .= '<div class="include"'.$this->_showTagLogos().'>'.DOKU_LF; 130f9f65fc3Swikidesign if (!$this->hasheader && $this->clevel && ($this->mode == 'section')) 131b6337848Swikidesign $renderer->doc .= '<div class="level'.$this->clevel.'">'.DOKU_LF; 132*8d99695dSwikidesign if ((@file_exists(DOKU_PLUGIN.'editsections/action.php')) 133*8d99695dSwikidesign && (!plugin_isdisabled('editsections'))){ // for Edit Section Reorganizer Plugin 1346cad866dSwikidesign $renderer->doc .= $this->_editButton().$content.DOKU_LF; 1356cad866dSwikidesign } else { 136b6337848Swikidesign $renderer->doc .= $content.DOKU_LF.$this->_editButton(); 1376cad866dSwikidesign } 138f9f65fc3Swikidesign if (!$this->hasheader && $this->clevel && ($this->mode == 'section')) 139b6337848Swikidesign $renderer->doc .= '</div>'.DOKU_LF; 140b6337848Swikidesign $renderer->doc .= '</div>'.DOKU_LF; 141f9f65fc3Swikidesign 142b6337848Swikidesign // output meta line (if wanted) and remove page from filechain 143b6337848Swikidesign $renderer->doc .= $this->_metaLine(array_pop($this->pages), $renderer); 144f9f65fc3Swikidesign 145f9f65fc3Swikidesign return $this->doc; 146f9f65fc3Swikidesign } 147f9f65fc3Swikidesign 148f9f65fc3Swikidesign/* ---------- Private Methods ---------- */ 149f9f65fc3Swikidesign 150f9f65fc3Swikidesign /** 151f9f65fc3Swikidesign * Get a section including its subsections 152f9f65fc3Swikidesign */ 153f9f65fc3Swikidesign function _getSection(){ 154f9f65fc3Swikidesign foreach ($this->ins as $ins){ 155f9f65fc3Swikidesign if ($ins[0] == 'header'){ 156f9f65fc3Swikidesign 157f9f65fc3Swikidesign // found the right header 158f9f65fc3Swikidesign if (cleanID($ins[1][0]) == $this->page['section']){ 159f9f65fc3Swikidesign $level = $ins[1][1]; 160f9f65fc3Swikidesign $i[] = $ins; 161f9f65fc3Swikidesign 162f9f65fc3Swikidesign // next header of the same or higher level -> exit 163f9f65fc3Swikidesign } elseif ($ins[1][1] <= $level){ 164f9f65fc3Swikidesign $this->ins = $i; 165f9f65fc3Swikidesign return true; 166f9f65fc3Swikidesign } elseif (isset($level)){ 167f9f65fc3Swikidesign $i[] = $ins; 168f9f65fc3Swikidesign } 169f9f65fc3Swikidesign 170f9f65fc3Swikidesign // add instructions from our section 171f9f65fc3Swikidesign } elseif (isset($level)){ 172f9f65fc3Swikidesign $i[] = $ins; 173f9f65fc3Swikidesign } 174f9f65fc3Swikidesign } 175f9f65fc3Swikidesign $this->ins = $i; 176f9f65fc3Swikidesign return true; 177f9f65fc3Swikidesign } 178f9f65fc3Swikidesign 179f9f65fc3Swikidesign /** 180f9f65fc3Swikidesign * Corrects relative internal links and media and 181f9f65fc3Swikidesign * converts headers of included pages to subheaders of the current page 182f9f65fc3Swikidesign */ 183f9f65fc3Swikidesign function _convertInstructions(&$renderer){ 184f9f65fc3Swikidesign global $ID; 185f9f65fc3Swikidesign global $conf; 186f9f65fc3Swikidesign 187f9f65fc3Swikidesign // check if included page is in same namespace 188f9f65fc3Swikidesign $inclNS = getNS($this->page['id']); 189f9f65fc3Swikidesign if (getNS($ID) == $inclNS) $convert = false; 190f9f65fc3Swikidesign else $convert = true; 191f9f65fc3Swikidesign 192f9f65fc3Swikidesign $n = count($this->ins); 193f9f65fc3Swikidesign for ($i = 0; $i < $n; $i++){ 194f9f65fc3Swikidesign 195f9f65fc3Swikidesign // convert internal links and media from relative to absolute 196f9f65fc3Swikidesign if ($convert && (substr($this->ins[$i][0], 0, 8) == 'internal')){ 197f9f65fc3Swikidesign 198f9f65fc3Swikidesign // relative subnamespace 199f9f65fc3Swikidesign if ($this->ins[$i][1][0]{0} == '.'){ 200f9f65fc3Swikidesign // parent namespace 201f9f65fc3Swikidesign if ($this->ins[$i][1][0]{1} == '.') 202f9f65fc3Swikidesign $ithis->ns[$i][1][0] = getNS($inclNS).':'.substr($this->ins[$i][1][0], 2); 203f9f65fc3Swikidesign // current namespace 204f9f65fc3Swikidesign else 205f9f65fc3Swikidesign $this->ins[$i][1][0] = $inclNS.':'.substr($this->ins[$i][1][0], 1); 206f9f65fc3Swikidesign 207f9f65fc3Swikidesign // relative link 208f9f65fc3Swikidesign } elseif (strpos($this->ins[$i][1][0], ':') === false){ 209f9f65fc3Swikidesign $this->ins[$i][1][0] = $inclNS.':'.$this->ins[$i][1][0]; 210f9f65fc3Swikidesign } 211f9f65fc3Swikidesign 212f9f65fc3Swikidesign // set header level to current section level + header level 213f9f65fc3Swikidesign } elseif ($this->ins[$i][0] == 'header'){ 214f9f65fc3Swikidesign $level = $this->ins[$i][1][1] + $this->clevel; 215f9f65fc3Swikidesign if ($level > 5) $level = 5; 216f9f65fc3Swikidesign $this->ins[$i][1][1] = $level; 217f9f65fc3Swikidesign 218f9f65fc3Swikidesign // add TOC items 219f9f65fc3Swikidesign if (($level >= $conf['toptoclevel']) && ($level <= $conf['maxtoclevel'])){ 220f9f65fc3Swikidesign $text = $this->ins[$i][1][0]; 221f9f65fc3Swikidesign $hid = $renderer->_headerToLink($text, 'true'); 222f9f65fc3Swikidesign $renderer->toc[] = array( 223f9f65fc3Swikidesign 'hid' => $hid, 224f9f65fc3Swikidesign 'title' => $text, 225f9f65fc3Swikidesign 'type' => 'ul', 226f9f65fc3Swikidesign 'level' => $level - $conf['toptoclevel'] + 1 227f9f65fc3Swikidesign ); 228f9f65fc3Swikidesign 229f9f65fc3Swikidesign $this->hasheader = true; 230f9f65fc3Swikidesign } 231f9f65fc3Swikidesign 232f9f65fc3Swikidesign // the same for sections 233f9f65fc3Swikidesign } elseif ($this->ins[$i][0] == 'section_open'){ 234f9f65fc3Swikidesign $level = $this->ins[$i][1][0] + $this->clevel; 235f9f65fc3Swikidesign if ($level > 5) $level = 5; 236f9f65fc3Swikidesign $this->ins[$i][1][0] = $level; 237f9f65fc3Swikidesign 238f9f65fc3Swikidesign // show only the first section? 239f9f65fc3Swikidesign } elseif ($this->firstsec && ($this->ins[$i][0] == 'section_close') 240f9f65fc3Swikidesign && ($this->ins[$i-1][0] != 'section_open')){ 241f9f65fc3Swikidesign if ($this->ins[0][0] == 'document_start'){ 242f9f65fc3Swikidesign $this->ins = array_slice($this->ins, 1, $i); 243f9f65fc3Swikidesign return true; 244f9f65fc3Swikidesign } else { 245f9f65fc3Swikidesign $this->ins = array_slice($this->ins, 0, $i); 246f9f65fc3Swikidesign return true; 247f9f65fc3Swikidesign } 248f9f65fc3Swikidesign } 249f9f65fc3Swikidesign } 250f9f65fc3Swikidesign if ($this->ins[0][0] == 'document_start') $this->ins = array_slice($this->ins, 1, -1); 251f9f65fc3Swikidesign return true; 252f9f65fc3Swikidesign } 253f9f65fc3Swikidesign 254f9f65fc3Swikidesign /** 255f9f65fc3Swikidesign * Remove TOC, section edit buttons and tags 256f9f65fc3Swikidesign */ 257f9f65fc3Swikidesign function _cleanXHTML($xhtml){ 258f9f65fc3Swikidesign preg_match('!<div class="tags">.*?</div>!s', $xhtml, $match); 259f9f65fc3Swikidesign $this->page['tags'] = $match[0]; 260f9f65fc3Swikidesign $replace = array( 261f9f65fc3Swikidesign '!<div class="toc">.*?(</div>\n</div>)!s' => '', // remove toc 262f9f65fc3Swikidesign '#<!-- SECTION "(.*?)" \[(\d+-\d*)\] -->#e' => '', // remove section edit buttons 263f9f65fc3Swikidesign '!<div class="tags">.*?(</div>)!s' => '', // remove category tags 264f9f65fc3Swikidesign ); 265f9f65fc3Swikidesign $xhtml = preg_replace(array_keys($replace), array_values($replace), $xhtml); 266f9f65fc3Swikidesign return $xhtml; 267f9f65fc3Swikidesign } 268f9f65fc3Swikidesign 269f9f65fc3Swikidesign /** 270f9f65fc3Swikidesign * Optionally display logo for the first tag found in the included page 271f9f65fc3Swikidesign */ 272f9f65fc3Swikidesign function _showTagLogos(){ 273f9f65fc3Swikidesign if (!$this->getConf('showtaglogos')) return ''; 274f9f65fc3Swikidesign 275f9f65fc3Swikidesign preg_match_all('/<a [^>]*title="(.*?)" rel="tag"[^>]*>([^<]*)</', $this->page['tags'], $tag); 276f9f65fc3Swikidesign $logoID = getNS($tag[1][0]).':'.$tag[2][0]; 277f9f65fc3Swikidesign $logosrc = mediaFN($logoID); 278f9f65fc3Swikidesign $types = array('.png', '.jpg', '.gif'); // auto-detect filetype 279f9f65fc3Swikidesign foreach ($types as $type){ 280f9f65fc3Swikidesign if (!@file_exists($logosrc.$type)) continue; 281f9f65fc3Swikidesign $logoID .= $type; 282f9f65fc3Swikidesign $logosrc .= $type; 283f9f65fc3Swikidesign list($w, $h, $t, $a) = getimagesize($logosrc); 284f9f65fc3Swikidesign return ' style="min-height: '.$h.'px">'. 285f9f65fc3Swikidesign '<img class="mediaright" src="'.ml($logoID).'" alt="'.$tag[2][0].'"/'; 286f9f65fc3Swikidesign } 287f9f65fc3Swikidesign return ''; 288f9f65fc3Swikidesign } 289f9f65fc3Swikidesign 290f9f65fc3Swikidesign /** 291f9f65fc3Swikidesign * Display an edit button for the included page 292f9f65fc3Swikidesign */ 293f9f65fc3Swikidesign function _editButton(){ 294f9f65fc3Swikidesign if (!isset($this->page['perm'])) 295f9f65fc3Swikidesign $this->page['perm'] = auth_quickaclcheck($this->page['id']); 296f9f65fc3Swikidesign if (@file_exists($this->page['file'])){ 297f9f65fc3Swikidesign if (($this->page['perm'] >= AUTH_EDIT) && (is_writable($this->page['file']))) 298f9f65fc3Swikidesign $action = 'edit'; 299f9f65fc3Swikidesign else return ''; 300f9f65fc3Swikidesign } elseif ($this->page['perm'] >= AUTH_CREATE){ 301f9f65fc3Swikidesign $action = 'create'; 302f9f65fc3Swikidesign } 303f9f65fc3Swikidesign return '<div class="secedit">'.DOKU_LF.DOKU_TAB. 304f9f65fc3Swikidesign html_btn($action, $this->page['id'], '', array('do' => 'edit'), 'post').DOKU_LF. 305f9f65fc3Swikidesign '</div>'.DOKU_LF; 306f9f65fc3Swikidesign } 307f9f65fc3Swikidesign 308f9f65fc3Swikidesign /** 309f9f65fc3Swikidesign * Adds a read more... link at the bottom of the first section 310f9f65fc3Swikidesign */ 311f9f65fc3Swikidesign function _readMore(){ 312f9f65fc3Swikidesign $last = $this->ins[count($this->ins) - 1]; 313f9f65fc3Swikidesign if ($last[0] == 'section_close') $this->ins = array_slice($this->ins, 0, -1); 314f9f65fc3Swikidesign $this->ins[] = array('p_open', array(), $last[2]); 315f9f65fc3Swikidesign $this->ins[] = array('internallink', array($this->page['id'], $this->getLang('readmore')), $last[2]); 316f9f65fc3Swikidesign $this->ins[] = array('p_close', array(), $last[2]); 317f9f65fc3Swikidesign if ($last[0] == 'section_close') $this->ins[] = $last; 318f9f65fc3Swikidesign } 319f9f65fc3Swikidesign 320f9f65fc3Swikidesign /** 321f9f65fc3Swikidesign * Returns the meta line below the included page 322f9f65fc3Swikidesign */ 323b6337848Swikidesign function _metaLine($page, &$renderer){ 324f9f65fc3Swikidesign global $conf; 325f9f65fc3Swikidesign 326f9f65fc3Swikidesign if (!$this->getConf('showmetaline')) 327f9f65fc3Swikidesign return '<div class="inclmeta"> </div>'.DOKU_LF; 328f9f65fc3Swikidesign 329b6337848Swikidesign $id = $page['id']; 330f9f65fc3Swikidesign $meta = p_get_metadata($id); 331f9f65fc3Swikidesign $ret = array(); 332f9f65fc3Swikidesign 333f9f65fc3Swikidesign // permalink 334f9f65fc3Swikidesign if ($this->getConf('showlink')){ 335b6337848Swikidesign $title = ($page['title'] ? $page['title'] : $meta['title']); 336f9f65fc3Swikidesign if (!$title) $title = str_replace('_', ' ', noNS($id)); 337f9f65fc3Swikidesign $ret[] = $renderer->internallink($id, $title, '', true); 338f9f65fc3Swikidesign } 339f9f65fc3Swikidesign 340f9f65fc3Swikidesign // date 341f9f65fc3Swikidesign if ($this->getConf('showdate')){ 342b6337848Swikidesign $date = ($page['date'] ? $page['date'] : $meta['date']['created']); 343e4624397Swikidesign if ($date) $ret[] = date($conf['dformat'], $date); 344f9f65fc3Swikidesign } 345f9f65fc3Swikidesign 346f9f65fc3Swikidesign // author 347f9f65fc3Swikidesign if ($this->getConf('showuser')){ 348b6337848Swikidesign $author = ($page['user'] ? $page['user'] : $meta['creator']); 349f9f65fc3Swikidesign if ($author){ 350f9f65fc3Swikidesign $userpage = cleanID($this->getConf('usernamespace').':'.$author); 351f9f65fc3Swikidesign $ret[] = $renderer->internallink($userpage, $author, '', true); 352f9f65fc3Swikidesign } 353f9f65fc3Swikidesign } 354f9f65fc3Swikidesign 355f9f65fc3Swikidesign // comments - let Discussion Plugin do the work for us 356b6337848Swikidesign if (!$page['section'] && $this->getConf('showcomments') 357e4624397Swikidesign && (!plugin_isdisabled('discussion')) 358e4624397Swikidesign && ($discussion =& plugin_load('helper', 'discussion'))){ 359e4624397Swikidesign $disc = $discussion->td($id); 360e4624397Swikidesign if ($disc) $ret[] = $disc; 361f9f65fc3Swikidesign } 362f9f65fc3Swikidesign 363f9f65fc3Swikidesign $ret = implode(' · ', $ret); 364f9f65fc3Swikidesign 365f9f65fc3Swikidesign // tags 366b6337848Swikidesign if (($this->getConf('showtags')) && ($page['tags'])){ 367f9f65fc3Swikidesign $ret = $this->page['tags'].$ret; 368f9f65fc3Swikidesign } 369f9f65fc3Swikidesign 370f9f65fc3Swikidesign if (!$ret) $ret = ' '; 371f9f65fc3Swikidesign return '<div class="inclmeta">'.DOKU_LF.$ret.DOKU_LF.'</div>'.DOKU_LF; 372f9f65fc3Swikidesign } 373f9f65fc3Swikidesign 374f9f65fc3Swikidesign} 375f9f65fc3Swikidesign 376f9f65fc3Swikidesign//Setup VIM: ex: et ts=4 enc=utf-8 : 377