1c677cb9fSwikidesign<?php 2c677cb9fSwikidesign/** 3c677cb9fSwikidesign * Include Plugin: Display a wiki page within another wiki page 4c677cb9fSwikidesign * 5c677cb9fSwikidesign * Action plugin component, for cache validity determination 6c677cb9fSwikidesign * 7c677cb9fSwikidesign * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 8c677cb9fSwikidesign * @author Christopher Smith <chris@jalakai.co.uk> 961053b04SMichael Klier * @author Michael Klier <chi@chimeric.de> 10c677cb9fSwikidesign */ 11c677cb9fSwikidesignif(!defined('DOKU_INC')) die(); // no Dokuwiki, no go 12c677cb9fSwikidesign 13c677cb9fSwikidesignif(!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/'); 14c677cb9fSwikidesignrequire_once(DOKU_PLUGIN.'action.php'); 15c677cb9fSwikidesign 16c677cb9fSwikidesign/** 17c677cb9fSwikidesign * All DokuWiki plugins to extend the parser/rendering mechanism 18c677cb9fSwikidesign * need to inherit from this class 19c677cb9fSwikidesign */ 20c677cb9fSwikidesignclass action_plugin_include extends DokuWiki_Action_Plugin { 21c677cb9fSwikidesign 22791e1550SMichael Hamann var $supportedModes = array('xhtml', 'metadata'); 2361053b04SMichael Klier var $helper = null; 2461053b04SMichael Klier 2561053b04SMichael Klier function action_plugin_include() { 2661053b04SMichael Klier $this->helper = plugin_load('helper', 'include'); 2761053b04SMichael Klier } 28c677cb9fSwikidesign 29c677cb9fSwikidesign /** 30c677cb9fSwikidesign * plugin should use this method to register its handlers with the dokuwiki's event controller 31c677cb9fSwikidesign */ 32c677cb9fSwikidesign function register(&$controller) { 33c677cb9fSwikidesign $controller->register_hook('PARSER_CACHE_USE','BEFORE', $this, '_cache_prepare'); 34c44026dcSMichael Hamann $controller->register_hook('HTML_EDITFORM_OUTPUT', 'BEFORE', $this, 'handle_form'); 35c44026dcSMichael Hamann $controller->register_hook('HTML_CONFLICTFORM_OUTPUT', 'BEFORE', $this, 'handle_form'); 36c44026dcSMichael Hamann $controller->register_hook('HTML_DRAFTFORM_OUTPUT', 'BEFORE', $this, 'handle_form'); 37c44026dcSMichael Hamann $controller->register_hook('ACTION_SHOW_REDIRECT', 'BEFORE', $this, 'handle_redirect'); 386f0ad9d7SMichael Klier $controller->register_hook('PARSER_HANDLER_DONE', 'BEFORE', $this, 'handle_parser'); 397488c4d8SMichael Klier $controller->register_hook('PARSER_METADATA_RENDER', 'AFTER', $this, 'handle_metadata'); 40993fccf3SMichael Hamann $controller->register_hook('HTML_SECEDIT_BUTTON', 'BEFORE', $this, 'handle_secedit_button'); 417488c4d8SMichael Klier } 427488c4d8SMichael Klier 437488c4d8SMichael Klier /** 447488c4d8SMichael Klier * Used for debugging purposes only 457488c4d8SMichael Klier */ 467488c4d8SMichael Klier function handle_metadata(&$event, $param) { 477488c4d8SMichael Klier global $conf; 487488c4d8SMichael Klier if($conf['allowdebug']) { 497488c4d8SMichael Klier dbglog('---- PLUGIN INCLUDE META DATA START ----'); 507488c4d8SMichael Klier dbglog($event->data); 517488c4d8SMichael Klier dbglog('---- PLUGIN INCLUDE META DATA END ----'); 527488c4d8SMichael Klier } 5361053b04SMichael Klier } 5461053b04SMichael Klier 5561053b04SMichael Klier /** 5661053b04SMichael Klier * Supplies the current section level to the include syntax plugin 5761053b04SMichael Klier * 5861053b04SMichael Klier * @author Michael Klier <chi@chimeric.de> 591a25f14bSMichael Hamann * @author Michael Hamann <michael@content-space.de> 6061053b04SMichael Klier */ 6161053b04SMichael Klier function handle_parser(&$event, $param) { 6261053b04SMichael Klier global $ID; 6361053b04SMichael Klier 64e14109dfSMichael Hamann $level = 0; 6561053b04SMichael Klier $ins =& $event->data->calls; 6661053b04SMichael Klier $num = count($ins); 6761053b04SMichael Klier for($i=0; $i<$num; $i++) { 681a25f14bSMichael Hamann switch($ins[$i][0]) { 691a25f14bSMichael Hamann case 'plugin': 70e5a4d2cdSMichael Klier switch($ins[$i][1][0]) { 71e5a4d2cdSMichael Klier case 'include_include': 721a25f14bSMichael Hamann $ins[$i][1][1][] = $level; 73e5a4d2cdSMichael Klier break; 741a25f14bSMichael Hamann /* FIXME: this doesn't work anymore that way with the new structure 75e5a4d2cdSMichael Klier // some plugins already close open sections 76e5a4d2cdSMichael Klier // so we need to make sure we don't close them twice 77e5a4d2cdSMichael Klier case 'box': 78e5a4d2cdSMichael Klier $this->helper->sec_close = false; 79e5a4d2cdSMichael Klier break; 801a25f14bSMichael Hamann */ 81e5a4d2cdSMichael Klier } 821a25f14bSMichael Hamann break; 831a25f14bSMichael Hamann case 'section_open': 841a25f14bSMichael Hamann $level = $ins[$i][1][0]; 851a25f14bSMichael Hamann break; 8661053b04SMichael Klier } 8761053b04SMichael Klier } 88c44026dcSMichael Hamann } 89c44026dcSMichael Hamann 90c44026dcSMichael Hamann /** 916f0ad9d7SMichael Klier * Add a hidden input to the form to preserve the redirect_id 92c44026dcSMichael Hamann */ 93c44026dcSMichael Hamann function handle_form(&$event, $param) { 94c44026dcSMichael Hamann if (array_key_exists('redirect_id', $_REQUEST)) { 95c44026dcSMichael Hamann $event->data->addHidden('redirect_id', cleanID($_REQUEST['redirect_id'])); 96c44026dcSMichael Hamann } 97c44026dcSMichael Hamann } 98c44026dcSMichael Hamann 99c44026dcSMichael Hamann /** 1006f0ad9d7SMichael Klier * Modify the data for the redirect when there is a redirect_id set 101c44026dcSMichael Hamann */ 102c44026dcSMichael Hamann function handle_redirect(&$event, $param) { 103c44026dcSMichael Hamann if (array_key_exists('redirect_id', $_REQUEST)) { 104dd385353SMichael Hamann // Render metadata when this is an older DokuWiki version where 105dd385353SMichael Hamann // metadata is not automatically re-rendered as the page has probably 106dd385353SMichael Hamann // been changed but is not directly displayed 107dd385353SMichael Hamann $versionData = getVersionData(); 108dd385353SMichael Hamann if ($versionData['date'] < '2010-11-23') { 109dd385353SMichael Hamann p_set_metadata($event->data['id'], array(), true); 110dd385353SMichael Hamann } 111c44026dcSMichael Hamann $event->data['id'] = cleanID($_REQUEST['redirect_id']); 112c44026dcSMichael Hamann $event->data['title'] = ''; 113c44026dcSMichael Hamann } 114c677cb9fSwikidesign } 115c677cb9fSwikidesign 116c677cb9fSwikidesign /** 117c677cb9fSwikidesign * prepare the cache object for default _useCache action 118c677cb9fSwikidesign */ 119c677cb9fSwikidesign function _cache_prepare(&$event, $param) { 1206f0ad9d7SMichael Klier global $conf; 1216f0ad9d7SMichael Klier 122c677cb9fSwikidesign $cache =& $event->data; 123c677cb9fSwikidesign 124791e1550SMichael Hamann if(!isset($cache->page)) return; 125c677cb9fSwikidesign if(!isset($cache->mode) || !in_array($cache->mode, $this->supportedModes)) return; 126c677cb9fSwikidesign 127791e1550SMichael Hamann $depends = p_get_metadata($cache->page, 'plugin_include'); 1287488c4d8SMichael Klier 1297488c4d8SMichael Klier if($conf['allowdebug']) { 1307488c4d8SMichael Klier dbglog('---- PLUGIN INCLUDE CACHE DEPENDS START ----'); 1317488c4d8SMichael Klier dbglog($depends); 1327488c4d8SMichael Klier dbglog('---- PLUGIN INCLUDE CACHE DEPENDS END ----'); 1337488c4d8SMichael Klier } 1347488c4d8SMichael Klier 135791e1550SMichael Hamann if (!is_array($depends)) return; // nothing to do for us 136c677cb9fSwikidesign 137791e1550SMichael Hamann if (!is_array($depends['pages']) || 138791e1550SMichael Hamann !is_array($depends['instructions']) || 13912092bceSMichael Hamann $depends['pages'] != $this->helper->_get_included_pages_from_meta_instructions($depends['instructions']) || 14012092bceSMichael Hamann // the include_content url parameter may change the behavior for included pages 141b11b2b9dSMichael Hamann $depends['include_content'] != isset($_REQUEST['include_content'])) { 142791e1550SMichael Hamann 143791e1550SMichael Hamann $cache->depends['purge'] = true; // included pages changed or old metadata - request purge. 144791e1550SMichael Hamann if($conf['allowdebug']) { 145791e1550SMichael Hamann dbglog('---- PLUGIN INCLUDE: REQUESTING CACHE PURGE ----'); 146791e1550SMichael Hamann dbglog('---- PLUGIN INCLUDE CACHE PAGES FROM META START ----'); 147791e1550SMichael Hamann dbglog($depends['pages']); 148791e1550SMichael Hamann dbglog('---- PLUGIN INCLUDE CACHE PAGES FROM META END ----'); 149791e1550SMichael Hamann dbglog('---- PLUGIN INCLUDE CACHE PAGES FROM META_INSTRUCTIONS START ----'); 150791e1550SMichael Hamann dbglog($this->helper->_get_included_pages_from_meta_instructions($depends['instructions'])); 151791e1550SMichael Hamann dbglog('---- PLUGIN INCLUDE CACHE PAGES FROM META_INSTRUCTIONS END ----'); 152791e1550SMichael Hamann 153791e1550SMichael Hamann } 154791e1550SMichael Hamann } else { 155526b0f2eSMichael Klier // add plugin.info.txt to depends for nicer upgrades 156526b0f2eSMichael Klier $cache->depends['files'][] = dirname(__FILE__) . '/plugin.info.txt'; 157791e1550SMichael Hamann foreach ($depends['pages'] as $page) { 158791e1550SMichael Hamann if (!$page['exists']) continue; 159791e1550SMichael Hamann $file = wikiFN($page['id']); 160791e1550SMichael Hamann if (!in_array($file, $cache->depends['files'])) { 1616f0ad9d7SMichael Klier $cache->depends['files'][] = $file; 1626f0ad9d7SMichael Klier } 1636f0ad9d7SMichael Klier } 1641a25f14bSMichael Hamann } 165c677cb9fSwikidesign } 166c677cb9fSwikidesign 16756aef185SMichael Hamann /** 16856aef185SMichael Hamann * Handle special section edit buttons for the include plugin to get the current page 16956aef185SMichael Hamann * and replace normal section edit buttons when the current page is different from the 17056aef185SMichael Hamann * global $ID. 17156aef185SMichael Hamann */ 172993fccf3SMichael Hamann function handle_secedit_button(&$event, $params) { 17356aef185SMichael Hamann // stack of included pages in the form ('id' => page, 'rev' => modification time, 'writable' => bool) 174993fccf3SMichael Hamann static $page_stack = array(); 175993fccf3SMichael Hamann 176993fccf3SMichael Hamann global $ID; 177993fccf3SMichael Hamann 178993fccf3SMichael Hamann $data = $event->data; 179993fccf3SMichael Hamann 180dcec2f1fSMichael Hamann if ($data['target'] == 'plugin_include_start' || $data['target'] == 'plugin_include_start_noredirect') { 181993fccf3SMichael Hamann // handle the "section edits" added by the include plugin 182993fccf3SMichael Hamann $fn = wikiFN($data['name']); 183993fccf3SMichael Hamann array_unshift($page_stack, array( 184993fccf3SMichael Hamann 'id' => $data['name'], 185993fccf3SMichael Hamann 'rev' => @filemtime($fn), 186dcec2f1fSMichael Hamann 'writable' => (is_writable($fn) && auth_quickaclcheck($data['name']) >= AUTH_EDIT), 187dcec2f1fSMichael Hamann 'redirect' => ($data['target'] == 'plugin_include_start'), 188993fccf3SMichael Hamann )); 189993fccf3SMichael Hamann } elseif ($data['target'] == 'plugin_include_end') { 190993fccf3SMichael Hamann array_shift($page_stack); 191993fccf3SMichael Hamann } elseif (!empty($page_stack)) { 192*02ca71c3SMichael Hamann if ($page_stack[0]['writable'] && isset($data['name']) && $data['name'] !== '') { 193993fccf3SMichael Hamann $name = $data['name']; 194993fccf3SMichael Hamann unset($data['name']); 195993fccf3SMichael Hamann 196993fccf3SMichael Hamann $secid = $data['secid']; 197993fccf3SMichael Hamann unset($data['secid']); 198993fccf3SMichael Hamann 199dcec2f1fSMichael Hamann if ($page_stack[0]['redirect']) 200993fccf3SMichael Hamann $data['redirect_id'] = $ID; 201993fccf3SMichael Hamann 202993fccf3SMichael Hamann $event->result = "<div class='secedit editbutton_" . $data['target'] . 203993fccf3SMichael Hamann " editbutton_" . $secid . "'>" . 204993fccf3SMichael Hamann html_btn('secedit', $page_stack[0]['id'], '', 205993fccf3SMichael Hamann array_merge(array('do' => 'edit', 206993fccf3SMichael Hamann 'rev' => $page_stack[0]['rev'], 207993fccf3SMichael Hamann 'summary' => '['.$name.'] '), $data), 208993fccf3SMichael Hamann 'post', $name) . '</div>'; 209993fccf3SMichael Hamann } else { 210993fccf3SMichael Hamann $event->result = ''; 211993fccf3SMichael Hamann } 212993fccf3SMichael Hamann } else { 213993fccf3SMichael Hamann return; // return so the event won't be stopped 214993fccf3SMichael Hamann } 215993fccf3SMichael Hamann 216993fccf3SMichael Hamann $event->preventDefault(); 217993fccf3SMichael Hamann $event->stopPropagation(); 218993fccf3SMichael Hamann } 219c677cb9fSwikidesign} 2202524d407SMichael Hamann// vim:ts=4:sw=4:et: 221