* @author Christopher Smith * @author Gina Häußge, Michael Klier */ // must be run within Dokuwiki if (!defined('DOKU_INC')) die(); if (!defined('DOKU_LF')) define('DOKU_LF', "\n"); if (!defined('DOKU_TAB')) define('DOKU_TAB', "\t"); if (!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN', DOKU_INC.'lib/plugins/'); class helper_plugin_include extends DokuWiki_Plugin { // DokuWiki_Helper_Plugin var $includes = array(); var $toplevel_id = NULL; var $toplevel = NULL; var $defaults = array(); /** * Constructor loads default config settings once */ function helper_plugin_include() { $this->defaults['firstsec'] = $this->getConf('firstseconly'); $this->defaults['editbtn'] = $this->getConf('showeditbtn'); $this->defaults['taglogos'] = $this->getConf('showtaglogos'); $this->defaults['footer'] = $this->getConf('showfooter'); $this->defaults['redirect'] = $this->getConf('doredirect'); $this->defaults['date'] = $this->getConf('showdate'); $this->defaults['user'] = $this->getConf('showuser'); $this->defaults['comments'] = $this->getConf('showcomments'); $this->defaults['linkbacks'] = $this->getConf('linkbacks'); $this->defaults['tags'] = $this->getConf('tags'); $this->defaults['link'] = $this->getConf('showlink'); } function getInfo() { return array( 'author' => 'Gina Häußge, Michael Klier, Esther Brunner', 'email' => 'dokuwiki@chimeric.de', 'date' => @file_get_contents(DOKU_PLUGIN . 'blog/VERSION'), 'name' => 'Include Plugin (helper class)', 'desc' => 'Functions to include another page in a wiki page', 'url' => 'http://wiki.splitbrain.org/plugin:include', ); } /** * Available methods for other plugins */ function getMethods() { $result = array(); $result[] = array( 'name' => 'get_flags', 'desc' => 'overrides standard values for showfooter and firstseconly settings', 'params' => array('flags' => 'array'), ); return $result; } /** * Overrides standard values for showfooter and firstseconly settings */ function get_flags($setflags) { // load defaults $flags = array(); $flags = $this->defaults; foreach ($setflags as $flag) { switch ($flag) { case 'footer': $flags['footer'] = 1; break; case 'nofooter': $flags['footer'] = 0; break; case 'firstseconly': case 'firstsectiononly': $flags['firstsec'] = 1; break; case 'fullpage': $flags['firstsec'] = 0; break; case 'noheader': $flags['noheader'] = 1; break; case 'editbtn': case 'editbutton': $flags['editbtn'] = 1; break; case 'noeditbtn': case 'noeditbutton': $flags['editbtn'] = 0; break; case 'permalink': $flags['permalink'] = 1; break; case 'nopermalink': $flags['permalink'] = 0; break; case 'redirect': $flags['redirect'] = 1; break; case 'noredirect': $flags['redirect'] = 0; break; } } return $flags; } /** * Parses the instructions list of the page which contains the includes * * @author Michael Klier */ function parse_instructions($id, &$ins) { $num = count($ins); $lvl = false; $prev_lvl = false; $mode = ''; $page = ''; $flags = array(); $range = false; for($i=0; $i<$num; $i++) { // set current level if($ins[$i][0] == 'section_open') { $lvl = $ins[$i][1][0]; if($i > $range) $prev_lvl = $lvl; } if($ins[$i][0] == 'plugin' && $ins[$i][1][0] == 'include_include' ) { // found no previous section set lvl to 0 if(!$lvl) $lvl = 0; // check if toplevel is set already if(!isset($this->toplevel)) $this->toplevel = $lvl; $mode = $ins[$i][1][1][0]; $page = $ins[$i][1][1][1]; $sect = $ins[$i][1][1][2]; $flags = $ins[$i][1][1][3]; // check if we left the range of possible sub includes and reset lvl to toplevel if($range && ($i > $range)) { if($prev_lvl) { $lvl = $prev_lvl; $prev_lvl = false; } else { $lvl = $this->toplevel; } } $page = $this->_apply_macro($page); resolve_pageid(getNS($id), $page, $exists); // resolve shortcuts $flags = $this->get_flags($flags); $ins_inc = $this->_get_instructions($page, $sect, $mode, $lvl, $flags); if(!empty($ins_inc)) { // combine instructions and reset counter $ins_start = array_slice($ins, 0, $i+1); $ins_end = array_slice($ins, $i+1); $range = $i + count($ins_inc); $ins = array_merge($ins_start, $ins_inc, $ins_end); $num = count($ins); } } } } /** * Returns the converted instructions of a give page/section * * @author Michael Klier */ function _get_instructions($page, $sect, $mode, $lvl, $flags) { global $ID; if($ID == $page || !page_exists($page) || (page_exists($page) && auth_quickaclcheck($page) < AUTH_READ)) return array(); $key = (!$sect) ? $page . '#' . $sect : $page; // prevent recursion if(!$this->includes[$key]) { $ins = p_cached_instructions(wikiFN($page)); $this->includes[$key] = true; $this->_convert_instructions($ins, $lvl, $page, $sect, $flags); return $ins; } } /** * Converts instructions of the included page * * The funcion iterates over the given list of instructions and generates * an index of header and section indicies. It also removes document * start/end instructions, converts links, and removes unwanted * instructions like tags, comments, linkbacks. * * Later all header/section levels are convertet to match the current * inclusion level. * * @author Michael Klier */ function _convert_instructions(&$ins, $lvl, $page, $sect, $flags) { // filter instructions if needed if(!empty($sect)) { $this->_get_section($ins, $sect); // section required } if($flags['firstsec']) { $this->_get_firstsec($ins, $page); // only first section } $ns = getNS($page); $num = count($ins); $conv_idx = array(); // conversion index $lvl_max = false; // max level $first_header = -1; $no_header = false; for($i=0; $i<$num; $i++) { switch($ins[$i][0]) { case 'document_start': case 'document_end': case 'section_edit': unset($ins[$i]); break; case 'header': // check if we need to skip the first header if((!$no_header) && $flags['noheader']) { unset($ins[$i]); $no_header = true; continue; } $conv_idx[] = $i; // get index of first header if($first_header == -1) $first_header = $i; // get max level if this instructions set if(!$lvl_max || ($ins[$i][1][1] < $lvl_max)) { $lvl_max = $ins[$i][1][1]; } break; case 'section_open': $conv_idx[] = $i; break; case 'internallink': case 'internalmedia': if($ins[$i][1][0]{0} == '.') { if($ins[$i][1][0]{1} == '.') { $ins[$i][1][0] = getNS($ns) . ':' . substr($ins[$i][1][0], 2); // parent namespace } else { $ins[$i][1][0] = $ns . ':' . substr($ins[$i][1][0], 1); // current namespace } } elseif (strpos($ins[$i][1][0], ':') === false) { $ins[$i][1][0] = $ns . ':' . $ins[$i][1][0]; // relative links } break; case 'plugin': // FIXME skip other plugins? if($ins[$i][1][0] == 'tag_tag') unset($ins[$i]); // skip tags if($ins[$i][1][0] == 'discussion_comments') unset($ins[$i]); // skip comments if($ins[$i][1][0] == 'linkback') unset($ins[$i]); // skip linkbacks break; default: break; } } // calculate difference between header/section level and include level $diff = 0; if($lvl_max) { // max level equals inclusion level diff is 1 if($lvl_max == $lvl) { $diff = 1; } // max level is les than inclusion level, we have to convert downwards if($lvl_max < $lvl) { $diff = (($lvl - $lvl_max) + 1); } // max level is greate inclusion level, we have to convert upwards if($lvl_max > $lvl) { if($lvl == 0) { // we had no previous section pretend it was 1 $diff = (1 - $lvl_max); } elseif(($lvl - $lvl_max) == -1) { // we don't need to convert anything up diff is 0 $diff = 0; } else { // convert everything up $diff = ($lvl - $lvl_max); } } } // convert headers and set footer/permalink $has_permalink = false; $footer_lvl = false; foreach($conv_idx as $idx) { if($ins[$idx][0] == 'header') { $new_lvl = (($ins[$idx][1][1] + $diff) > 5) ? 5 : ($ins[$idx][1][1] + $diff); $ins[$idx][1][1] = $new_lvl; // get section title if we only include one section if(($sect && !$sect_title) && ($idx == $first_header)) { $sect_title = $ins[$idx][1][0]; } // set permalink if($flags['link'] && !$has_permalink && ($idx == $first_header)) { $this->_permalink($ins[$idx], $page, $sect, $flags); $has_permalink = true; } } else { // it's a section $new_lvl = (($ins[$idx][1][0] + $diff) > 5) ? 5 : ($ins[$idx][1][0] + $diff); $ins[$idx][1][0] = $new_lvl; // check if noheader is used and set the footer level to the first section if($no_header && !$footer_lvl) $footer_lvl = $new_lvl; } // set footer level if(!$footer_lvl && ($idx == $first_header)) $footer_lvl = $new_lvl; } // add edit button if($flags['editbtn']) $this->_editbtn($ins, $page, $sect, $sect_title); // add footer if($flags['footer']) $this->_footer($ins, $page, $sect, $sect_title, $flags, $footer_lvl); // add instructions entry divs array_unshift($ins, array('plugin', array('include_div', array('open', $page)))); array_push($ins, array('plugin', array('include_div', array('close')))); // close previous section if any and re-open after inclusion if($lvl != 0) { array_unshift($ins, array('section_close')); $ins[] = array('section_open', array($lvl)); } } /** * Appends instruction item for the include plugin footer * * @author Michael Klier */ function _footer(&$ins, $page, $sect, $sect_title, $flags, $footer_lvl) { $footer = array(); $footer[0] = 'plugin'; $footer[1] = array('include_footer', array($page, $sect, $sect_title, $flags, $this->toplevel_id, $footer_lvl)); $ins[] = $footer; } /** * Appends instruction item for an edit button * * @author Michael Klier */ function _editbtn(&$ins, $page, $sect, $sect_title) { $editbtn = array(); $editbtn[0] = 'plugin'; $editbtn[1] = array('include_editbtn', array($page, $sect, $sect_title, $this->toplevel_id)); $ins[] = $editbtn; } /** * Convert instruction item for a permalink header * * @author Michael Klier */ function _permalink(&$ins, $page, $sect, $flags) { $ins[0] = 'plugin'; $ins[1] = array('include_header', array($ins[1][0], $ins[1][1], $page, $sect, $flags)); } /** * Get a section including its subsections * * @author Michael Klier */ function _get_section(&$ins, $sect) { $num = count($ins); $offset = false; $lvl = false; for($i=0; $i<$num; $i++) { if ($ins[$i][0] == 'header') { // found the right header if (cleanID($ins[$i][1][0]) == $sect) { $offset = $i; $lvl = $ins[$i][1][1]; } elseif ($offset && $lvl && ($ins[$i][1][1] <= $lvl)) { $ins = array_slice($ins, $offset, ($i - $offset)); } } } } /** * Only display the first section of a page and a readmore link * * @author Michael Klier */ function _get_firstsec(&$ins, $page) { $num = count($ins); for($i=0; $i<$num; $i++) { if($ins[$i][0] == 'section_close') { $ins = array_slice($ins, 0, $i); $ins[] = array('p_open', array()); $ins[] = array('internallink',array($page, $this->getLang('readmore'))); $ins[] = array('p_close', array()); $ins[] = array('section_close'); return; } } } /** * Makes user or date dependent includes possible */ function _apply_macro($id) { global $INFO; global $auth; // if we don't have an auth object, do nothing if (!$auth) return $id; $user = $_SERVER['REMOTE_USER']; $userdata = $auth->getUserData($user); $group = $userdata['grps'][0]; $replace = array( '@USER@' => cleanID($user), '@NAME@' => cleanID($INFO['userinfo']['name']), '@GROUP@' => cleanID($group), '@YEAR@' => date('Y'), '@MONTH@' => date('m'), '@DAY@' => date('d'), ); return str_replace(array_keys($replace), array_values($replace), $id); } } //vim:ts=4:sw=4:et:enc=utf-8: