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 226f0ad9d7SMichael Klier var $supportedModes = array('xhtml', 'i'); 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 * return some info 31c677cb9fSwikidesign */ 32c677cb9fSwikidesign function getInfo() { 33c677cb9fSwikidesign return array( 34dbdadbd9SGina Haeussge 'author' => 'Gina Häußge, Michael Klier, Christopher Smith', 35dbdadbd9SGina Haeussge 'email' => 'dokuwiki@chimeric.de', 364052f233SGina Haeussge 'date' => @file_get_contents(DOKU_PLUGIN . 'blog/VERSION'), 37c677cb9fSwikidesign 'name' => 'Include Plugin', 38c44026dcSMichael Hamann 'desc' => 'Improved cache handling for included pages and redirect-handling', 395013e176SMichael Klier 'url' => 'http://dokuwiki.org/plugin:include', 40c677cb9fSwikidesign ); 41c677cb9fSwikidesign } 42c677cb9fSwikidesign 43c677cb9fSwikidesign /** 44c677cb9fSwikidesign * plugin should use this method to register its handlers with the dokuwiki's event controller 45c677cb9fSwikidesign */ 46c677cb9fSwikidesign function register(&$controller) { 47c677cb9fSwikidesign $controller->register_hook('PARSER_CACHE_USE','BEFORE', $this, '_cache_prepare'); 4861053b04SMichael Klier// $controller->register_hook('PARSER_CACHE_USE','AFTER', $this, '_cache_result'); // debugging only 49c44026dcSMichael Hamann $controller->register_hook('HTML_EDITFORM_OUTPUT', 'BEFORE', $this, 'handle_form'); 50c44026dcSMichael Hamann $controller->register_hook('HTML_CONFLICTFORM_OUTPUT', 'BEFORE', $this, 'handle_form'); 51c44026dcSMichael Hamann $controller->register_hook('HTML_DRAFTFORM_OUTPUT', 'BEFORE', $this, 'handle_form'); 52c44026dcSMichael Hamann $controller->register_hook('ACTION_SHOW_REDIRECT', 'BEFORE', $this, 'handle_redirect'); 536f0ad9d7SMichael Klier $controller->register_hook('PARSER_HANDLER_DONE', 'BEFORE', $this, 'handle_parser'); 5461053b04SMichael Klier } 5561053b04SMichael Klier 5661053b04SMichael Klier /** 5761053b04SMichael Klier * Supplies the current section level to the include syntax plugin 5861053b04SMichael Klier * 5961053b04SMichael Klier * @author Michael Klier <chi@chimeric.de> 6061053b04SMichael Klier */ 6161053b04SMichael Klier function handle_parser(&$event, $param) { 6261053b04SMichael Klier global $ID; 6361053b04SMichael Klier 646f0ad9d7SMichael Klier // check for stored toplevel ID in helper plugin 656f0ad9d7SMichael Klier // if it's missing lets see if we have to do anything at all 666f0ad9d7SMichael Klier if(!isset($this->helper->toplevel_id)) { 6761053b04SMichael Klier $ins =& $event->data->calls; 6861053b04SMichael Klier $num = count($ins); 6961053b04SMichael Klier for($i=0; $i<$num; $i++) { 706f0ad9d7SMichael Klier if(($ins[$i][0] == 'plugin') && ($ins[$i][1][0] == 'include_include')) { 714d3c3e05SMichael Klier if(!isset($this->helper->toplevel_id)) $this->helper->toplevel_id = $ID; 726f0ad9d7SMichael Klier $this->helper->parse_instructions($ID, $ins); 7361053b04SMichael Klier } 7461053b04SMichael Klier } 7561053b04SMichael Klier } 76c44026dcSMichael Hamann } 77c44026dcSMichael Hamann 78c44026dcSMichael Hamann /** 796f0ad9d7SMichael Klier * Add a hidden input to the form to preserve the redirect_id 80c44026dcSMichael Hamann */ 81c44026dcSMichael Hamann function handle_form(&$event, $param) { 82c44026dcSMichael Hamann if (array_key_exists('redirect_id', $_REQUEST)) { 83c44026dcSMichael Hamann $event->data->addHidden('redirect_id', cleanID($_REQUEST['redirect_id'])); 84c44026dcSMichael Hamann } 85c44026dcSMichael Hamann } 86c44026dcSMichael Hamann 87c44026dcSMichael Hamann /** 886f0ad9d7SMichael Klier * Modify the data for the redirect when there is a redirect_id set 89c44026dcSMichael Hamann */ 90c44026dcSMichael Hamann function handle_redirect(&$event, $param) { 91c44026dcSMichael Hamann if (array_key_exists('redirect_id', $_REQUEST)) { 92c44026dcSMichael Hamann $event->data['id'] = cleanID($_REQUEST['redirect_id']); 93c44026dcSMichael Hamann $event->data['title'] = ''; 94c44026dcSMichael Hamann } 95c677cb9fSwikidesign } 96c677cb9fSwikidesign 97c677cb9fSwikidesign /** 98c677cb9fSwikidesign * prepare the cache object for default _useCache action 99c677cb9fSwikidesign */ 100c677cb9fSwikidesign function _cache_prepare(&$event, $param) { 1016f0ad9d7SMichael Klier global $ID; 1026f0ad9d7SMichael Klier global $conf; 1036f0ad9d7SMichael Klier 104c677cb9fSwikidesign $cache =& $event->data; 105c677cb9fSwikidesign 1066f0ad9d7SMichael Klier // we're only interested in instructions of the current page 1076f0ad9d7SMichael Klier // without the ID check we'd get the cache objects for included pages as well 1086f0ad9d7SMichael Klier if(!isset($cache->page) && ($cache->page != $ID)) return; 109c677cb9fSwikidesign if(!isset($cache->mode) || !in_array($cache->mode, $this->supportedModes)) return; 110c677cb9fSwikidesign 1116f0ad9d7SMichael Klier // get additional depends 1126f0ad9d7SMichael Klier $depends = p_get_metadata($ID, 'relation haspart'); 1136f0ad9d7SMichael Klier if(empty($depends)) return; 114c677cb9fSwikidesign 1156f0ad9d7SMichael Klier $key = ''; 1166f0ad9d7SMichael Klier foreach(array_keys($depends) as $page) { 1176f0ad9d7SMichael Klier if(strpos($page,'/') || cleanID($page) != $page) { 1186f0ad9d7SMichael Klier continue; 1196f0ad9d7SMichael Klier } else { 1206f0ad9d7SMichael Klier $file = wikiFN($page); 1216f0ad9d7SMichael Klier if(!in_array($cache->depends['files'], array($file)) && @file_exists($file)) { 1226f0ad9d7SMichael Klier $cache->depends['files'][] = $file; 123*298f626bSMichael Klier $key .= '#' . $page . '|ACL' . auth_quickaclcheck($page); 1246f0ad9d7SMichael Klier } 1256f0ad9d7SMichael Klier } 1266f0ad9d7SMichael Klier } 127c677cb9fSwikidesign 128c677cb9fSwikidesign // empty $key implies no includes, so nothing to do 129c677cb9fSwikidesign if(empty($key)) return; 130c677cb9fSwikidesign 131c677cb9fSwikidesign // mark the cache as being modified by the include plugin 132c677cb9fSwikidesign $cache->include = true; 133c677cb9fSwikidesign 1346f0ad9d7SMichael Klier // set new cache key & cache name 1356f0ad9d7SMichael Klier // now also dependent on included page ids and their ACL_READ status 136c677cb9fSwikidesign $cache->key .= $key; 137c677cb9fSwikidesign $cache->cache = getCacheName($cache->key, $cache->ext); 138c677cb9fSwikidesign } 139c677cb9fSwikidesign 140c677cb9fSwikidesign function _cache_result(&$event, $param) { 141c677cb9fSwikidesign $cache =& $event->data; 142c677cb9fSwikidesign if (empty($cache->include)) return; 14361053b04SMichael Klier //global $debug; 14461053b04SMichael Klier //$debug['cache_result'][] = $event->result ? 'true' : 'false'; 145c677cb9fSwikidesign } 146c677cb9fSwikidesign} 147df4e907bSMichael Klier//vim:ts=4:sw=4:et:enc=utf-8: 148