xref: /plugin/addnewpage/syntax.php (revision 89aa939b046e5dda5f0b2d9f782da1f17c6578f1)
15ad1ba23SRobert McLeod<?php
2*89aa939bSDamien Regad
3c2f9de42SSam Wilson/**
4c2f9de42SSam Wilson * Add-New-Page Plugin: a simple form for adding new pages.
5c2f9de42SSam Wilson *
6c2f9de42SSam Wilson * @license  GPL 2 (http://www.gnu.org/licenses/gpl.html)
7c2f9de42SSam Wilson * @author   iDO <ido@idotech.info>
8c2f9de42SSam Wilson * @author   Sam Wilson <sam@samwilson.id.au>
97c0b9c8bSDamien Regad *
100aa2c115SDamien Regad * @noinspection PhpUnused
110aa2c115SDamien Regad * @noinspection PhpMissingParamTypeInspection, PhpMissingReturnTypeInspection
12c2f9de42SSam Wilson */
130102fe8cSDamien Regad
1418fb4458SDamien Regaduse dokuwiki\Extension\SyntaxPlugin;
1551a509d7SDamien Regaduse dokuwiki\File\PageResolver;
1651a509d7SDamien Regad
17*89aa939bSDamien Regadclass syntax_plugin_addnewpage extends SyntaxPlugin
18*89aa939bSDamien Regad{
19b8304a8eSAndreas Gohr    /** @var array the parsed options */
20b8304a8eSAndreas Gohr    protected $options;
21b8304a8eSAndreas Gohr
22c2f9de42SSam Wilson    /**
23d49bfc3cSGerrit Uitslag     * Syntax Type
24c2f9de42SSam Wilson     */
25*89aa939bSDamien Regad    public function getType()
26*89aa939bSDamien Regad    {
2758a4331aSGerrit Uitslag        return 'substition';
2858a4331aSGerrit Uitslag    }
295ad1ba23SRobert McLeod
30d49bfc3cSGerrit Uitslag    /**
31d49bfc3cSGerrit Uitslag     * Paragraph Type
32d49bfc3cSGerrit Uitslag     */
33*89aa939bSDamien Regad    public function getPType()
34*89aa939bSDamien Regad    {
3558a4331aSGerrit Uitslag        return 'block';
3658a4331aSGerrit Uitslag    }
375ad1ba23SRobert McLeod
38d49bfc3cSGerrit Uitslag    /**
39d49bfc3cSGerrit Uitslag     * @return int
40d49bfc3cSGerrit Uitslag     */
41*89aa939bSDamien Regad    public function getSort()
42*89aa939bSDamien Regad    {
4358a4331aSGerrit Uitslag        return 199;
4458a4331aSGerrit Uitslag    }
455ad1ba23SRobert McLeod
46d49bfc3cSGerrit Uitslag    /**
47d49bfc3cSGerrit Uitslag     * @param string $mode
48d49bfc3cSGerrit Uitslag     */
49*89aa939bSDamien Regad    public function connectTo($mode)
50*89aa939bSDamien Regad    {
515ad1ba23SRobert McLeod        $this->Lexer->addSpecialPattern('\{\{NEWPAGE[^\}]*\}\}', $mode, 'plugin_addnewpage');
525ad1ba23SRobert McLeod    }
535ad1ba23SRobert McLeod
54d49bfc3cSGerrit Uitslag    /**
55d2f65217SDamien Regad     * Handler to prepare matched data for the rendering process.
56d49bfc3cSGerrit Uitslag     *
57735f6285SGerrit Uitslag     * Handled syntax options:
58d2f65217SDamien Regad     * - {{NEWPAGE}}
59d2f65217SDamien Regad     * - {{NEWPAGE>your:namespace}}
60d2f65217SDamien Regad     * - {{NEWPAGE>your:namespace:@INPUT@:start}}
61d2f65217SDamien Regad     * - {{NEWPAGE>your:namespace:[date formats]}} {@see strftime()}
62d2f65217SDamien Regad     * - {{NEWPAGE?config_overrides}}
632c279077SDamien Regad     * - {{NEWPAGE?label=custom}}
64d2f65217SDamien Regad     * - {{NEWPAGE#newtpl1,newtpl2}}
65d2f65217SDamien Regad     * - {{NEWPAGE#newtpl1|Title1,newtpl2|Title1}}
66d2f65217SDamien Regad     * - {{NEWPAGE>your:namespace#newtpl1|Title1,newtpl2|Title1}}
67d2f65217SDamien Regad     * - {{NEWPAGE>your:namespace#newtpl1|Title1,newtpl2|Title1#@HI@,Howdy}}
68d2f65217SDamien Regad     *
69d2f65217SDamien Regad     * Refer to {@see https://www.dokuwiki.org/plugin:addnewpage} for details.
70735f6285SGerrit Uitslag     *
71d49bfc3cSGerrit Uitslag     * @param   string $match The text matched by the patterns
72d49bfc3cSGerrit Uitslag     * @param   int $state The lexer state for the match
73d49bfc3cSGerrit Uitslag     * @param   int $pos The character position of the matched text
74d49bfc3cSGerrit Uitslag     * @param   Doku_Handler $handler The Doku_Handler object
75d2f65217SDamien Regad     *
76d49bfc3cSGerrit Uitslag     * @return  array Return an array with all data you want to use in render
7703475bb4SSam Wilson     * @codingStandardsIgnoreStart
78d49bfc3cSGerrit Uitslag     */
79716e610aSGerrit Uitslag    public function handle($match, $state, $pos, Doku_Handler $handler) {
8003475bb4SSam Wilson        /* @codingStandardsIgnoreEnd */
81b8304a8eSAndreas Gohr        $match = substr($match, 9, -2); // strip markup
82716e610aSGerrit Uitslag
83*89aa939bSDamien Regad        $data = [
84b8304a8eSAndreas Gohr            'namespace' => '',
85*89aa939bSDamien Regad            'newpagetemplates' => [],
8631967e04SDamien Regad            'newpagevars' => '',
87*89aa939bSDamien Regad            'options' => [
88b8304a8eSAndreas Gohr                'exclude' => $this->getConf('addpage_exclude'),
89b8304a8eSAndreas Gohr                'showroot' => $this->getConf('addpage_showroot'),
90b8304a8eSAndreas Gohr                'hide' => $this->getConf('addpage_hide'),
91b8304a8eSAndreas Gohr                'hideacl' => $this->getConf('addpage_hideACL'),
92b8304a8eSAndreas Gohr                'autopage' => $this->getConf('addpage_autopage'),
935187ea72SDamien Regad                'createns' => $this->getConf('addpage_createns'),
942c279077SDamien Regad                'label' => 'okbutton',
95*89aa939bSDamien Regad            ]
96*89aa939bSDamien Regad        ];
97b8304a8eSAndreas Gohr
98b8304a8eSAndreas Gohr        if (preg_match('/>(.*?)(#|\?|$)/', $match, $m)) {
99b8304a8eSAndreas Gohr            $data['namespace'] = trim($m[1]);
100b8304a8eSAndreas Gohr        }
101b8304a8eSAndreas Gohr
102d49ba5bbSDamien Regad        # Extract the newpagetemplate plugin parameters
103d49ba5bbSDamien Regad        # - after the initial #: the template name
104d49ba5bbSDamien Regad        # - after optional 2nd #: custom variable names
105d49ba5bbSDamien Regad        if (preg_match('/#(.*?)(?:#(.*?))?(?:\?|$)/', $match, $m)) {
106b8304a8eSAndreas Gohr            $data['newpagetemplates'] = array_map('trim', explode(',', $m[1]));
107d49ba5bbSDamien Regad            $data['newpagevars'] = trim($m[2] ?? '');
108b8304a8eSAndreas Gohr        }
109b8304a8eSAndreas Gohr
110b8304a8eSAndreas Gohr        if (preg_match('/\?(.*?)(#|$)/', $match, $m)) {
111*89aa939bSDamien Regad            $this->parseOptions($m[1], $data['options']);
112b8304a8eSAndreas Gohr        }
113b8304a8eSAndreas Gohr
114b8304a8eSAndreas Gohr        return $data;
1155ad1ba23SRobert McLeod    }
1165ad1ba23SRobert McLeod
117c2f9de42SSam Wilson    /**
118c2f9de42SSam Wilson     * Create the new-page form.
119c2f9de42SSam Wilson     *
1207c0b9c8bSDamien Regad     * @param   $format   string        output format being rendered
121d49bfc3cSGerrit Uitslag     * @param   $renderer Doku_Renderer the current renderer object
122d49bfc3cSGerrit Uitslag     * @param   $data     array         data created by handler()
123d49bfc3cSGerrit Uitslag     * @return  boolean                 rendered correctly?
124c2f9de42SSam Wilson     */
125*89aa939bSDamien Regad    public function render($format, Doku_Renderer $renderer, $data)
126*89aa939bSDamien Regad    {
1275ad1ba23SRobert McLeod        global $lang;
1285ad1ba23SRobert McLeod
129b8304a8eSAndreas Gohr        // make options available in class
130b8304a8eSAndreas Gohr        $this->options = $data['options'];
131b8304a8eSAndreas Gohr
1327c0b9c8bSDamien Regad        if ($format == 'xhtml') {
13399f48d03SDamien Regad            $disablecache = false;
134*89aa939bSDamien Regad            $namespaceinput = $this->htmlNamespaceInput($data['namespace'], $disablecache);
135716e610aSGerrit Uitslag            if ($namespaceinput === false) {
136b8304a8eSAndreas Gohr                if ($this->options['hideacl']) {
137716e610aSGerrit Uitslag                    $renderer->doc .= '';
138716e610aSGerrit Uitslag                } else {
139716e610aSGerrit Uitslag                    $renderer->doc .= $this->getLang('nooption');
140716e610aSGerrit Uitslag                }
1415ad1ba23SRobert McLeod                return true;
1425ad1ba23SRobert McLeod            }
14399f48d03SDamien Regad            if ($disablecache) $renderer->info['cache'] = false;
1445ad1ba23SRobert McLeod
145*89aa939bSDamien Regad            $newpagetemplateinput = $this->htmlTemplateInput($data['newpagetemplates']);
146716e610aSGerrit Uitslag
147b8304a8eSAndreas Gohr            $input = 'text';
148b8304a8eSAndreas Gohr            if ($this->options['autopage']) $input = 'hidden';
149b8304a8eSAndreas Gohr
1502c279077SDamien Regad            // Button label. If given string is not localized, use it as-is
1512c279077SDamien Regad            $label = $this->getLang($this->options['label']);
1522c279077SDamien Regad            if (!$label) {
1532c279077SDamien Regad                $label = $this->options['label'];
1542c279077SDamien Regad            }
1552c279077SDamien Regad
156ed55cb4cSMichael Grosse            $form = '<div class="addnewpage"><p>'
157df9a92a0SDamien Regad                . '<form name="addnewpage" method="get" action="' . DOKU_BASE . DOKU_SCRIPT
158df9a92a0SDamien Regad                    . '" accept-charset="' . $lang['encoding'] . '">'
159ed55cb4cSMichael Grosse                . $namespaceinput
160*89aa939bSDamien Regad                . '<input class="edit" type="' . $input
161*89aa939bSDamien Regad                    . '" name="title" size="20" maxlength="255" tabindex="2" placeholder="'
1625187ea72SDamien Regad                    . $this->getLang('name')
1635187ea72SDamien Regad                    // Use a data attribute to pass the createns option's state to JavaScript
1645187ea72SDamien Regad                    . '" data-createns="' . $this->options['createns']
1655187ea72SDamien Regad                    . '"/>'
166716e610aSGerrit Uitslag                . $newpagetemplateinput
167ed55cb4cSMichael Grosse                . '<input type="hidden" name="newpagevars" value="' . $data['newpagevars'] . '"/>'
168ed55cb4cSMichael Grosse                . '<input type="hidden" name="do" value="edit" />'
169ed55cb4cSMichael Grosse                . '<input type="hidden" name="id" />'
1702c279077SDamien Regad                . '<input class="button" type="submit" value="' . $label . '" tabindex="4" />'
171ed55cb4cSMichael Grosse                . '</form>'
172aa8cc19bSMichael Grosse                . '</p></div>';
173ed55cb4cSMichael Grosse
174c2f9de42SSam Wilson            $renderer->doc .= $form;
1755ad1ba23SRobert McLeod
1765ad1ba23SRobert McLeod            return true;
1775ad1ba23SRobert McLeod        }
1785ad1ba23SRobert McLeod        return false;
1795ad1ba23SRobert McLeod    }
180c2f9de42SSam Wilson
1815ad1ba23SRobert McLeod    /**
182b8304a8eSAndreas Gohr     * Overwrites the $options with the ones parsed from $optstr
183b8304a8eSAndreas Gohr     *
184b8304a8eSAndreas Gohr     * @param string $optstr
185b8304a8eSAndreas Gohr     * @param array $options
186b8304a8eSAndreas Gohr     * @author Andreas Gohr <gohr@cosmocode.de>
187b8304a8eSAndreas Gohr     */
188*89aa939bSDamien Regad    protected function parseOptions($optstr, &$options)
189*89aa939bSDamien Regad    {
190b8304a8eSAndreas Gohr        $opts = preg_split('/[,&]/', $optstr);
191b8304a8eSAndreas Gohr
192b8304a8eSAndreas Gohr        foreach ($opts as $opt) {
19309b1853cSDamien Regad            $opt_lower = strtolower(trim($opt));
194b8304a8eSAndreas Gohr            $val = true;
195b8304a8eSAndreas Gohr            // booleans can be negated with a no prefix
19609b1853cSDamien Regad            if (substr($opt_lower, 0, 2) == 'no') {
19709b1853cSDamien Regad                $opt_lower = substr($opt, 2);
198b8304a8eSAndreas Gohr                $val = false;
199b8304a8eSAndreas Gohr            }
200b8304a8eSAndreas Gohr
201b8304a8eSAndreas Gohr            // not a known option? might be a key=value pair
20209b1853cSDamien Regad            if (!isset($options[$opt_lower])) {
20309b1853cSDamien Regad                $split = array_map('trim', sexplode('=', $opt, 2));
20409b1853cSDamien Regad                $opt_lower = strtolower($split[0]);
20509b1853cSDamien Regad                $val = $split[1];
206b8304a8eSAndreas Gohr            }
207b8304a8eSAndreas Gohr
208b8304a8eSAndreas Gohr            // still unknown? skip it
20909b1853cSDamien Regad            if (!isset($options[$opt_lower])) continue;
210b8304a8eSAndreas Gohr
211b8304a8eSAndreas Gohr            // overwrite the current value
21209b1853cSDamien Regad            $options[$opt_lower] = $val;
213b8304a8eSAndreas Gohr        }
214b8304a8eSAndreas Gohr    }
215b8304a8eSAndreas Gohr
216b8304a8eSAndreas Gohr    /**
2175ad1ba23SRobert McLeod     * Parse namespace request
2185ad1ba23SRobert McLeod     *
219b8304a8eSAndreas Gohr     * This creates the final ID to be created (still having an @INPUT@ variable
220b8304a8eSAndreas Gohr     * which is filled in via JavaScript)
221b8304a8eSAndreas Gohr     *
2225ad1ba23SRobert McLeod     * @author Samuele Tognini <samuele@cli.di.unipi.it>
223a98ad79cSSam Wilson     * @author Michael Braun <michael-dev@fami-braun.de>
224b8304a8eSAndreas Gohr     * @author Andreas Gohr <gohr@cosmocode.de>
225b8304a8eSAndreas Gohr     * @param string $ns The namespace as given in the syntax
226b8304a8eSAndreas Gohr     * @return string
2275ad1ba23SRobert McLeod     */
228*89aa939bSDamien Regad    protected function parseNS($ns)
229*89aa939bSDamien Regad    {
230fec857f5SMichael Große        global $INFO;
231b8304a8eSAndreas Gohr
232b8304a8eSAndreas Gohr        $selfid = $INFO['id'];
2330462edaaSDamien Regad        $selfns = getNS($selfid);
234b8304a8eSAndreas Gohr        // replace the input variable with something unique that survives cleanID
235b8304a8eSAndreas Gohr        $keep = sha1(time());
236b8304a8eSAndreas Gohr
237b8304a8eSAndreas Gohr        // by default append the input to the namespace (except on autopage)
238b8304a8eSAndreas Gohr        if (strpos($ns, '@INPUT@') === false && !$this->options['autopage']) $ns .= ":@INPUT@";
239b8304a8eSAndreas Gohr
240b8304a8eSAndreas Gohr        // date replacements
241b8304a8eSAndreas Gohr        $ns = dformat(null, $ns);
242b8304a8eSAndreas Gohr
243b8304a8eSAndreas Gohr        // placeholders
244*89aa939bSDamien Regad        $replacements = [
245b8304a8eSAndreas Gohr            '/\//' => ':', // forward slashes to colons
246b8304a8eSAndreas Gohr            '/@PAGE@/' => $selfid,
247b8304a8eSAndreas Gohr            '/@NS@/' => $selfns,
248b8304a8eSAndreas Gohr            '/^\.(:|\/|$)/' => "$selfns:",
249b8304a8eSAndreas Gohr            '/@INPUT@/' => $keep,
250*89aa939bSDamien Regad        ];
251b8304a8eSAndreas Gohr        $ns = preg_replace(array_keys($replacements), array_values($replacements), $ns);
252b8304a8eSAndreas Gohr
253b8304a8eSAndreas Gohr        // clean up, then reinsert the input variable
2545ad1ba23SRobert McLeod        $ns = cleanID($ns);
2550aa2c115SDamien Regad        return str_replace($keep, '@INPUT@', $ns);
2565ad1ba23SRobert McLeod    }
257c2f9de42SSam Wilson
258c2f9de42SSam Wilson    /**
259c2f9de42SSam Wilson     * Create the HTML Select element for namespace selection.
260c2f9de42SSam Wilson     *
2618fad05fbSSam Wilson     * @param string|false $dest_ns The destination namespace, or false if none provided.
262716e610aSGerrit Uitslag     * @param bool $disablecache reference indicates if caching need to be disabled
263716e610aSGerrit Uitslag     * @global string $ID The page ID
264c2f9de42SSam Wilson     * @return string Select element with appropriate NS selected.
265c2f9de42SSam Wilson     */
266*89aa939bSDamien Regad    protected function htmlNamespaceInput($dest_ns, &$disablecache)
267*89aa939bSDamien Regad    {
2685ad1ba23SRobert McLeod        global $ID;
269716e610aSGerrit Uitslag        $disablecache = false;
2705ad1ba23SRobert McLeod
2718fad05fbSSam Wilson        // If a NS has been provided:
2728fad05fbSSam Wilson        // Whether to hide the NS selection (otherwise, show only subnamespaces).
273b8304a8eSAndreas Gohr        $hide = $this->options['hide'];
2745ad1ba23SRobert McLeod
275*89aa939bSDamien Regad        $parsed_dest_ns = $this->parseNS($dest_ns);
2768fad05fbSSam Wilson        // Whether the user can create pages in the provided NS (or root, if no
2778fad05fbSSam Wilson        // destination NS has been set.
27829fe948bSJulian Einwag        $can_create = (auth_quickaclcheck($parsed_dest_ns . ":") >= AUTH_CREATE);
2798fad05fbSSam Wilson
280716e610aSGerrit Uitslag        //namespace given, but hidden
281716e610aSGerrit Uitslag        if ($hide && !empty($dest_ns)) {
2828fad05fbSSam Wilson            if ($can_create) {
28329fe948bSJulian Einwag                return '<input type="hidden" name="np_cat" id="np_cat" value="' . $parsed_dest_ns . '"/>';
2848fad05fbSSam Wilson            } else {
285716e610aSGerrit Uitslag                return false;
2868fad05fbSSam Wilson            }
287c2f9de42SSam Wilson        }
2885ad1ba23SRobert McLeod
289716e610aSGerrit Uitslag        //show select of given namespace
290716e610aSGerrit Uitslag        $currentns = getNS($ID);
2915ad1ba23SRobert McLeod
2925ad1ba23SRobert McLeod        $ret = '<select class="edit" id="np_cat" name="np_cat" tabindex="1">';
2935ad1ba23SRobert McLeod
2948fad05fbSSam Wilson        // Whether the NS select element has any options
2955ad1ba23SRobert McLeod        $someopt = false;
2965ad1ba23SRobert McLeod
2978fad05fbSSam Wilson        // Show root namespace if requested and allowed
298b8304a8eSAndreas Gohr        if ($this->options['showroot'] && $can_create) {
2998fad05fbSSam Wilson            if (empty($dest_ns)) {
3008fad05fbSSam Wilson                // If no namespace has been provided, add an option for the root NS.
301*89aa939bSDamien Regad                $ret .= '<option ' . (($currentns == '') ? 'selected ' : '')
302*89aa939bSDamien Regad                    . ' value="">' . $this->getLang('namespaceRoot') . '</option>';
3035ad1ba23SRobert McLeod            } else {
3048fad05fbSSam Wilson                // If a namespace has been provided, add an option for it.
305*89aa939bSDamien Regad                $ret .= '<option ' . (($currentns == $dest_ns) ? 'selected ' : '')
306*89aa939bSDamien Regad                    . ' value="' . formText($dest_ns) . '">' . formText($dest_ns) . '</option>';
3075ad1ba23SRobert McLeod            }
3087c0b9c8bSDamien Regad            $someopt = true;
309c2f9de42SSam Wilson        }
3105ad1ba23SRobert McLeod
311*89aa939bSDamien Regad        $subnamespaces = $this->getNamespaceList($dest_ns);
312fead6b5eSAlbert Chern
313fead6b5eSAlbert Chern        // The top of this stack will always be the last printed ancestor namespace
314*89aa939bSDamien Regad        $ancestor_stack = [];
315fead6b5eSAlbert Chern        if (!empty($dest_ns)) {
3167c0b9c8bSDamien Regad            $ancestor_stack[] = $dest_ns;
317fead6b5eSAlbert Chern        }
318fead6b5eSAlbert Chern
319716e610aSGerrit Uitslag        foreach ($subnamespaces as $ns) {
320716e610aSGerrit Uitslag            if (auth_quickaclcheck($ns . ":") < AUTH_CREATE) continue;
321fead6b5eSAlbert Chern
322fead6b5eSAlbert Chern            // Pop any elements off the stack that are not ancestors of the current namespace
323*89aa939bSDamien Regad            while ($ancestor_stack !== [] && strpos($ns, $ancestor_stack[count($ancestor_stack) - 1] . ':') !== 0) {
324fead6b5eSAlbert Chern                array_pop($ancestor_stack);
325fead6b5eSAlbert Chern            }
326fead6b5eSAlbert Chern
327716e610aSGerrit Uitslag            $nsparts = explode(':', $ns);
328*89aa939bSDamien Regad            $first_unprinted_depth = $ancestor_stack === []
329*89aa939bSDamien Regad                ? 1
330*89aa939bSDamien Regad                : (2 + substr_count($ancestor_stack[count($ancestor_stack) - 1], ':'));
331fead6b5eSAlbert Chern            for ($i = $first_unprinted_depth, $end = count($nsparts); $i <= $end; $i++) {
332fead6b5eSAlbert Chern                $namespace = implode(':', array_slice($nsparts, 0, $i));
3337c0b9c8bSDamien Regad                $ancestor_stack[] = $namespace;
334fead6b5eSAlbert Chern                $selectOptionText = str_repeat('&nbsp;&nbsp;', substr_count($namespace, ':')) . $nsparts[$i - 1];
335fead6b5eSAlbert Chern                $ret .= '<option ' .
336fead6b5eSAlbert Chern                    (($currentns == $namespace) ? 'selected ' : '') .
337fead6b5eSAlbert Chern                    ($i == $end ? ('value="' . $namespace . '">') : 'disabled>') .
338fead6b5eSAlbert Chern                    $selectOptionText .
339fead6b5eSAlbert Chern                    '</option>';
340fead6b5eSAlbert Chern            }
3415ad1ba23SRobert McLeod            $someopt = true;
342716e610aSGerrit Uitslag            $disablecache = true;
3435ad1ba23SRobert McLeod        }
344fead6b5eSAlbert Chern
3455ad1ba23SRobert McLeod        $ret .= '</select>';
3465ad1ba23SRobert McLeod
347716e610aSGerrit Uitslag        if ($someopt) {
3485ad1ba23SRobert McLeod            return $ret;
349716e610aSGerrit Uitslag        } else {
350716e610aSGerrit Uitslag            return false;
351716e610aSGerrit Uitslag        }
3525ad1ba23SRobert McLeod    }
353c2f9de42SSam Wilson
354f9dafbcdSSam Wilson    /**
355f9dafbcdSSam Wilson     * Get a list of namespaces below the given namespace.
356f9dafbcdSSam Wilson     * Recursively fetches subnamespaces.
357f9dafbcdSSam Wilson     *
358716e610aSGerrit Uitslag     * @param string $topns The top namespace
359f9dafbcdSSam Wilson     * @return array Multi-dimensional array of all namespaces below $tns
360f9dafbcdSSam Wilson     */
361*89aa939bSDamien Regad    protected function getNamespaceList($topns = '')
362*89aa939bSDamien Regad    {
3635ad1ba23SRobert McLeod        global $conf;
364d49bfc3cSGerrit Uitslag
365716e610aSGerrit Uitslag        $topns = utf8_encodeFN(str_replace(':', '/', $topns));
3665ad1ba23SRobert McLeod
367b8304a8eSAndreas Gohr        $excludes = $this->options['exclude'];
368716e610aSGerrit Uitslag        if ($excludes == "") {
369*89aa939bSDamien Regad            $excludes = [];
370716e610aSGerrit Uitslag        } else {
371716e610aSGerrit Uitslag            $excludes = @explode(';', strtolower($excludes));
372716e610aSGerrit Uitslag        }
373*89aa939bSDamien Regad        $searchdata = [];
374*89aa939bSDamien Regad        search($searchdata, $conf['datadir'], 'search_namespaces', [], $topns);
3755ad1ba23SRobert McLeod
376*89aa939bSDamien Regad        $namespaces = [];
377716e610aSGerrit Uitslag        foreach ($searchdata as $ns) {
378716e610aSGerrit Uitslag            foreach ($excludes as $exclude) {
379017bc029SMatthieu Estrada                if (!empty($exclude) && strpos($ns['id'], $exclude) === 0) {
380716e610aSGerrit Uitslag                    continue 2;
381716e610aSGerrit Uitslag                }
382716e610aSGerrit Uitslag            }
383716e610aSGerrit Uitslag            $namespaces[] = $ns['id'];
384716e610aSGerrit Uitslag        }
3855ad1ba23SRobert McLeod
386716e610aSGerrit Uitslag        return $namespaces;
3875ad1ba23SRobert McLeod    }
388716e610aSGerrit Uitslag
389716e610aSGerrit Uitslag    /**
390716e610aSGerrit Uitslag     * Create html for selection of namespace templates
391716e610aSGerrit Uitslag     *
392716e610aSGerrit Uitslag     * @param array $newpagetemplates array of namespace templates
393716e610aSGerrit Uitslag     * @return string html of select or hidden input
394716e610aSGerrit Uitslag     */
395*89aa939bSDamien Regad    public function htmlTemplateInput($newpagetemplates)
396*89aa939bSDamien Regad    {
397716e610aSGerrit Uitslag        $cnt = count($newpagetemplates);
398716e610aSGerrit Uitslag        if ($cnt < 1 || $cnt == 1 && $newpagetemplates[0] == '') {
399716e610aSGerrit Uitslag            $input = '';
400716e610aSGerrit Uitslag        } else {
401716e610aSGerrit Uitslag            if ($cnt == 1) {
402*89aa939bSDamien Regad                [$template, ] = $this->parseNSTemplatePage($newpagetemplates[0]);
403e6a74061SGerrit Uitslag                $input = '<input type="hidden" name="newpagetemplate" value="' . formText($template) . '" />';
404716e610aSGerrit Uitslag            } else {
405716e610aSGerrit Uitslag                $first = true;
4068c8f824eSGerrit Uitslag                $input = '<select name="newpagetemplate" tabindex="3">';
407716e610aSGerrit Uitslag                foreach ($newpagetemplates as $template) {
408716e610aSGerrit Uitslag                    $p = ($first ? ' selected="selected"' : '');
409716e610aSGerrit Uitslag                    $first = false;
410716e610aSGerrit Uitslag
411*89aa939bSDamien Regad                    [$template, $name] = $this->parseNSTemplatePage($template);
412716e610aSGerrit Uitslag                    $p .= ' value="' . formText($template) . '"';
413716e610aSGerrit Uitslag                    $input .= "<option $p>" . formText($name) . "</option>";
4145ad1ba23SRobert McLeod                }
415716e610aSGerrit Uitslag                $input .= '</select>';
4165ad1ba23SRobert McLeod            }
417716e610aSGerrit Uitslag            $input = DOKU_TAB . DOKU_TAB . $input . DOKU_LF;
4185ad1ba23SRobert McLeod        }
419716e610aSGerrit Uitslag        return $input;
420c2f9de42SSam Wilson    }
421716e610aSGerrit Uitslag
422716e610aSGerrit Uitslag    /**
423716e610aSGerrit Uitslag     * Parses and resolves the namespace template page
424716e610aSGerrit Uitslag     *
425716e610aSGerrit Uitslag     * @param $nstemplate
426716e610aSGerrit Uitslag     * @return array
427716e610aSGerrit Uitslag     */
428*89aa939bSDamien Regad    protected function parseNSTemplatePage($nstemplate)
429*89aa939bSDamien Regad    {
430716e610aSGerrit Uitslag        global $ID;
431716e610aSGerrit Uitslag
432*89aa939bSDamien Regad        @[$template, $name] = explode('|', $nstemplate, 2);
43351a509d7SDamien Regad        $template = (new PageResolver($ID))->resolveId($template);
434716e610aSGerrit Uitslag        if (is_null($name)) $name = $template;
435716e610aSGerrit Uitslag
436*89aa939bSDamien Regad        return [$template, $name];
4375ad1ba23SRobert McLeod    }
4385ad1ba23SRobert McLeod}
439