xref: /plugin/include/syntax/include.php (revision 1a25f14b28693cdcc54f732630b8204c660d5108)
171ec1101SGina Haeussge<?php
271ec1101SGina Haeussge/**
371ec1101SGina Haeussge * Include Plugin: displays a wiki page within another
471ec1101SGina Haeussge * Usage:
571ec1101SGina Haeussge * {{page>page}} for "page" in same namespace
671ec1101SGina Haeussge * {{page>:page}} for "page" in top namespace
771ec1101SGina Haeussge * {{page>namespace:page}} for "page" in namespace "namespace"
871ec1101SGina Haeussge * {{page>.namespace:page}} for "page" in subnamespace "namespace"
971ec1101SGina Haeussge * {{page>page#section}} for a section of "page"
1071ec1101SGina Haeussge *
1171ec1101SGina Haeussge * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
1271ec1101SGina Haeussge * @author     Esther Brunner <wikidesign@gmail.com>
1371ec1101SGina Haeussge * @author     Christopher Smith <chris@jalakai.co.uk>
1471ec1101SGina Haeussge * @author     Gina Häußge, Michael Klier <dokuwiki@chimeric.de>
1571ec1101SGina Haeussge */
1671ec1101SGina Haeussge
1771ec1101SGina Haeussgeif(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../../').'/');
1871ec1101SGina Haeussgeif(!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/');
1971ec1101SGina Haeussgerequire_once(DOKU_PLUGIN.'syntax.php');
2071ec1101SGina Haeussge
2171ec1101SGina Haeussge/**
2271ec1101SGina Haeussge * All DokuWiki plugins to extend the parser/rendering mechanism
2371ec1101SGina Haeussge * need to inherit from this class
2471ec1101SGina Haeussge */
2571ec1101SGina Haeussgeclass syntax_plugin_include_include extends DokuWiki_Syntax_Plugin {
2671ec1101SGina Haeussge
2761053b04SMichael Klier    var $helper = null;
28*1a25f14bSMichael Hamann    var $taghelper = null;
2961053b04SMichael Klier
3071ec1101SGina Haeussge    function getType() { return 'substition'; }
3171ec1101SGina Haeussge    function getSort() { return 303; }
3271ec1101SGina Haeussge    function getPType() { return 'block'; }
3371ec1101SGina Haeussge
3471ec1101SGina Haeussge    function connectTo($mode) {
3571ec1101SGina Haeussge        $this->Lexer->addSpecialPattern("{{page>.+?}}", $mode, 'plugin_include_include');
3671ec1101SGina Haeussge        $this->Lexer->addSpecialPattern("{{section>.+?}}", $mode, 'plugin_include_include');
378b99501bSMichael Klier        $this->Lexer->addSpecialPattern("{{namespace>.+?}}", $mode, 'plugin_include_include');
38a0a6f8fbSMichael Klier        $this->Lexer->addSpecialPattern("{{tagtopic>.+?}}", $mode, 'plugin_include_include');
3971ec1101SGina Haeussge    }
4071ec1101SGina Haeussge
4171ec1101SGina Haeussge    function handle($match, $state, $pos, &$handler) {
4271ec1101SGina Haeussge
4371ec1101SGina Haeussge        $match = substr($match, 2, -2); // strip markup
4471ec1101SGina Haeussge        list($match, $flags) = explode('&', $match, 2);
4571ec1101SGina Haeussge
4661053b04SMichael Klier        // break the pattern up into its parts
476f0ad9d7SMichael Klier        list($mode, $page, $sect) = preg_split('/>|#/u', $match, 3);
486f0ad9d7SMichael Klier        return array($mode, $page, cleanID($sect), explode('&', $flags));
4971ec1101SGina Haeussge    }
5071ec1101SGina Haeussge
51*1a25f14bSMichael Hamann    /**
52*1a25f14bSMichael Hamann     * Renders the included page(s)
53*1a25f14bSMichael Hamann     *
54*1a25f14bSMichael Hamann     * @author Michael Hamann <michael@content-space.de>
55*1a25f14bSMichael Hamann     */
5671ec1101SGina Haeussge    function render($format, &$renderer, $data) {
57*1a25f14bSMichael Hamann        global $ID, $conf;
58*1a25f14bSMichael Hamann
59*1a25f14bSMichael Hamann        // static stack that records all ancestors of the child pages
60*1a25f14bSMichael Hamann        static $page_stack = array();
61*1a25f14bSMichael Hamann
62*1a25f14bSMichael Hamann        // when there is no id just assume the global $ID is the current id
63*1a25f14bSMichael Hamann        if (empty($page_stack)) $page_stack[] = $ID;
64*1a25f14bSMichael Hamann
65*1a25f14bSMichael Hamann        $parent_id = $page_stack[count($page_stack)-1];
66*1a25f14bSMichael Hamann        $root_id = $page_stack[0];
67*1a25f14bSMichael Hamann
68*1a25f14bSMichael Hamann        list($mode, $page, $sect, $flags, $level) = $data;
69*1a25f14bSMichael Hamann
70*1a25f14bSMichael Hamann        if (!$this->helper)
71*1a25f14bSMichael Hamann            $this->helper =& plugin_load('helper', 'include');
72*1a25f14bSMichael Hamann        $flags = $this->helper->get_flags($flags);
73*1a25f14bSMichael Hamann
74*1a25f14bSMichael Hamann        $pages = array();
75*1a25f14bSMichael Hamann        switch($mode) {
76*1a25f14bSMichael Hamann        case 'namespace':
77*1a25f14bSMichael Hamann            $ns    = str_replace(':', '/', cleanID($page));
78*1a25f14bSMichael Hamann            search($pagearrays, $conf['datadir'], 'search_list', '', $ns);
79*1a25f14bSMichael Hamann            if (is_array($pagearrays)) {
80*1a25f14bSMichael Hamann                foreach ($pagearrays as $pagearray) {
81*1a25f14bSMichael Hamann                    $pages[] = $pagearray['id'];
82*1a25f14bSMichael Hamann                }
83*1a25f14bSMichael Hamann            }
84*1a25f14bSMichael Hamann            sort($pages);
85*1a25f14bSMichael Hamann            break;
86*1a25f14bSMichael Hamann        case 'tagtopic':
87*1a25f14bSMichael Hamann            if (!$this->taghelper)
88*1a25f14bSMichael Hamann                $this->taghelper =& plugin_load('helper', 'tag');
89*1a25f14bSMichael Hamann            if(!$this->taghelper) {
90*1a25f14bSMichael Hamann                msg('You have to install the tag plugin to use this functionality!', -1);
91*1a25f14bSMichael Hamann                return;
92*1a25f14bSMichael Hamann            }
93*1a25f14bSMichael Hamann            $tag   = $page;
94*1a25f14bSMichael Hamann            $sect  = '';
95*1a25f14bSMichael Hamann            $pagearrays = $this->taghelper->getTopic('', null, $tag);
96*1a25f14bSMichael Hamann            foreach ($pagearrays as $pagearray) {
97*1a25f14bSMichael Hamann                $pages[] = $pagearray['id'];
98*1a25f14bSMichael Hamann            }
99*1a25f14bSMichael Hamann            break;
100*1a25f14bSMichael Hamann        default:
101*1a25f14bSMichael Hamann            $page = cleanID($this->helper->_apply_macro($page));
102*1a25f14bSMichael Hamann            resolve_pageid(getNS($parent_id), $page, $exists); // resolve shortcuts
103*1a25f14bSMichael Hamann            $pages[] = $page;
104*1a25f14bSMichael Hamann        }
105*1a25f14bSMichael Hamann
106*1a25f14bSMichael Hamann        foreach ($pages as $page) {
107*1a25f14bSMichael Hamann            if (in_array($page, $page_stack)) continue;
108*1a25f14bSMichael Hamann            array_push($page_stack, $page);
109*1a25f14bSMichael Hamann
110*1a25f14bSMichael Hamann            if($format == 'metadata') {
111*1a25f14bSMichael Hamann                $renderer->meta['plugin_include']['pages'][] = $page; // FIXME: record raw id so caching can check if the current replacements still match
112*1a25f14bSMichael Hamann                // recording all included pages might make sense for metadata cache updates, though really knowing all included pages
113*1a25f14bSMichael Hamann                // probably means we just need to always purge the metadata cache or move rendered metadata to other cache files that depend on user/groups
114*1a25f14bSMichael Hamann                // for namespace/tag includes we probably need to purge the cache everytime so they should be recorded in the metadata so we know when that's necessary
115*1a25f14bSMichael Hamann            }
116*1a25f14bSMichael Hamann
117*1a25f14bSMichael Hamann            $perm = auth_quickaclcheck($page);
118*1a25f14bSMichael Hamann
119*1a25f14bSMichael Hamann            if($perm < AUTH_READ) continue;
120*1a25f14bSMichael Hamann
121*1a25f14bSMichael Hamann            if(!page_exists($page)) {
122*1a25f14bSMichael Hamann                if($flags['footer']) {
123*1a25f14bSMichael Hamann                    $renderer->nest(array($this->helper->_footer($page, $sect, '', $flags, $level, $root_id)));
124*1a25f14bSMichael Hamann                }
125*1a25f14bSMichael Hamann            } else {
126*1a25f14bSMichael Hamann                $instructions = $this->helper->_get_instructions($page, $sect, $mode, $level, $flags, $root_id);
127*1a25f14bSMichael Hamann                $renderer->nest($instructions);
128*1a25f14bSMichael Hamann            }
129*1a25f14bSMichael Hamann
130*1a25f14bSMichael Hamann            array_pop($page_stack);
131*1a25f14bSMichael Hamann        }
132*1a25f14bSMichael Hamann
133*1a25f14bSMichael Hamann        // When all includes have been handled remove the current id
134*1a25f14bSMichael Hamann        // in order to allow the rendering of other pages
135*1a25f14bSMichael Hamann        if (count($page_stack) == 1) array_pop($page_stack);
136*1a25f14bSMichael Hamann
137*1a25f14bSMichael Hamann        return true;
13861053b04SMichael Klier    }
13971ec1101SGina Haeussge}
14071ec1101SGina Haeussge// vim:ts=4:sw=4:et:enc=utf-8:
141