xref: /plugin/menu/syntax.php (revision a02b76109604eab25950751e965b6c8666055b15)
1dad308ecSroot<?php
2dad308ecSroot/**
3dad308ecSroot * Plugin: Displays a link list in a nice way
4dad308ecSroot *
5dad308ecSroot * Syntax: <menu col=2,align=center,caption="headline">
6dad308ecSroot *           <item>name|description|link|image</item>
7dad308ecSroot *         </menu>
8dad308ecSroot *
9dad308ecSroot * Options have to be separated by comma.
10dad308ecSroot * col (opt)     The number of columns of the menu. Allowed are 1-4, default is 1
11dad308ecSroot * align (opt)   Alignment of the menu. Allowed are "left", "center" or "right",
12dad308ecSroot *               default is "left"
13dad308ecSroot * caption (opt) Headline of the menu, default is none
14dad308ecSroot *
15dad308ecSroot * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
16dad308ecSroot * @author     Matthias Grimm <matthiasgrimm@users.sourceforge.net>
17dad308ecSroot * @author     Frank Schiebel <frank@linuxmuster.net>
18dad308ecSroot */
19dad308ecSroot
20dad308ecSrootif(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../../').'/');
21dad308ecSrootif(!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/');
22dad308ecSrootrequire_once(DOKU_PLUGIN.'syntax.php');
23dad308ecSroot
24dad308ecSroot/**
25dad308ecSroot * All DokuWiki plugins to extend the parser/rendering mechanism
26dad308ecSroot * need to inherit from this class
27dad308ecSroot */
28dad308ecSrootclass syntax_plugin_menu extends DokuWiki_Syntax_Plugin {
29dad308ecSroot
30dad308ecSroot    var $rcmd = array();  /**< Command array for the renderer */
31dad308ecSroot
32dad308ecSroot    function __construct() {
33dad308ecSroot    }
34dad308ecSroot
35dad308ecSroot
36dad308ecSroot   /**
37dad308ecSroot    * Get an associative array with plugin info.
38dad308ecSroot    *
39dad308ecSroot    * <p>
40dad308ecSroot    * The returned array holds the following fields:
41dad308ecSroot    * <dl>
42dad308ecSroot    * <dt>author</dt><dd>Author of the plugin</dd>
43dad308ecSroot    * <dt>email</dt><dd>Email address to contact the author</dd>
44dad308ecSroot    * <dt>date</dt><dd>Last modified date of the plugin in
45dad308ecSroot    * <tt>YYYY-MM-DD</tt> format</dd>
46dad308ecSroot    * <dt>name</dt><dd>Name of the plugin</dd>
47dad308ecSroot    * <dt>desc</dt><dd>Short description of the plugin (Text only)</dd>
48dad308ecSroot    * <dt>url</dt><dd>Website with more information on the plugin
49dad308ecSroot    * (eg. syntax description)</dd>
50dad308ecSroot    * </dl>
51dad308ecSroot    * @param none
52dad308ecSroot    * @return Array Information about this plugin class.
53dad308ecSroot    * @public
54dad308ecSroot    * @static
55dad308ecSroot    */
56dad308ecSroot    function getInfo(){
57dad308ecSroot        return array(
58dad308ecSroot            'author' => 'Matthias Grimm',
59dad308ecSroot            'email'  => 'matthiasgrimm@users.sourceforge.net',
60*a02b7610SEli Fenton            //'date'   => '2009-07-25',
61*a02b7610SEli Fenton			'date'   => '2018-11-27',
62dad308ecSroot            'name'   => 'Menu Plugin',
63dad308ecSroot            'desc'   => 'Shows a list of links as a nice menu card',
64*a02b7610SEli Fenton            'url'    => 'http://www.dokuwiki.org/plugin:menu',
65dad308ecSroot        );
66dad308ecSroot    }
67dad308ecSroot
68dad308ecSroot   /**
69dad308ecSroot    * Get the type of syntax this plugin defines.
70dad308ecSroot    *
71dad308ecSroot    * The type of this plugin is "protected". It has a start and an end
72dad308ecSroot    * token and no other wiki commands shall be parsed between them.
73dad308ecSroot    *
74dad308ecSroot    * @param none
75dad308ecSroot    * @return String <tt>'protected'</tt>.
76dad308ecSroot    * @public
77dad308ecSroot    * @static
78dad308ecSroot    */
79dad308ecSroot    function getType(){
80dad308ecSroot        return 'protected';
81dad308ecSroot    }
82dad308ecSroot
83dad308ecSroot   /**
84dad308ecSroot    * Define how this plugin is handled regarding paragraphs.
85dad308ecSroot    *
86dad308ecSroot    * <p>
87dad308ecSroot    * This method is important for correct XHTML nesting. It returns
88dad308ecSroot    * one of the following values:
89dad308ecSroot    * </p>
90dad308ecSroot    * <dl>
91dad308ecSroot    * <dt>normal</dt><dd>The plugin can be used inside paragraphs.</dd>
92dad308ecSroot    * <dt>block</dt><dd>Open paragraphs need to be closed before
93dad308ecSroot    * plugin output.</dd>
94dad308ecSroot    * <dt>stack</dt><dd>Special case: Plugin wraps other paragraphs.</dd>
95dad308ecSroot    * </dl>
96dad308ecSroot    * @param none
97dad308ecSroot    * @return String <tt>'block'</tt>.
98dad308ecSroot    * @public
99dad308ecSroot    * @static
100dad308ecSroot    */
101dad308ecSroot    function getPType(){
102dad308ecSroot        return 'block';
103dad308ecSroot    }
104dad308ecSroot
105dad308ecSroot   /**
106dad308ecSroot    * Where to sort in?
107dad308ecSroot    *
108dad308ecSroot    * Sort the plugin in just behind the formating tokens
109dad308ecSroot    *
110dad308ecSroot    * @param none
111dad308ecSroot    * @return Integer <tt>135</tt>.
112dad308ecSroot    * @public
113dad308ecSroot    * @static
114dad308ecSroot    */
115dad308ecSroot    function getSort(){
116dad308ecSroot        return 135;
117dad308ecSroot    }
118dad308ecSroot
119dad308ecSroot   /**
120dad308ecSroot    * Connect lookup pattern to lexer.
121dad308ecSroot    *
122dad308ecSroot    * @param $aMode String The desired rendermode.
123dad308ecSroot    * @return none
124dad308ecSroot    * @public
125dad308ecSroot    * @see render()
126dad308ecSroot    */
127dad308ecSroot    function connectTo($mode) {
128dad308ecSroot       $this->Lexer->addEntryPattern('<menu>(?=.*?</menu.*?>)',$mode,'plugin_menu');
129dad308ecSroot       $this->Lexer->addEntryPattern('<menu\s[^\r\n\|]*?>(?=.*?</menu.*?>)',$mode,'plugin_menu');
130dad308ecSroot    }
131dad308ecSroot
132dad308ecSroot    function postConnect() {
133dad308ecSroot      $this->Lexer->addPattern('<item>.+?</item>','plugin_menu');
134dad308ecSroot      $this->Lexer->addExitPattern('</menu>','plugin_menu');
135dad308ecSroot    }
136dad308ecSroot
137dad308ecSroot   /**
138dad308ecSroot    * Handler to prepare matched data for the rendering process.
139dad308ecSroot    *
140dad308ecSroot    * <p>
141dad308ecSroot    * The <tt>$aState</tt> parameter gives the type of pattern
142dad308ecSroot    * which triggered the call to this method:
143dad308ecSroot    * </p>
144dad308ecSroot    * <dl>
145dad308ecSroot    * <dt>DOKU_LEXER_ENTER</dt>
146dad308ecSroot    * <dd>a pattern set by <tt>addEntryPattern()</tt></dd>
147dad308ecSroot    * <dt>DOKU_LEXER_MATCHED</dt>
148dad308ecSroot    * <dd>a pattern set by <tt>addPattern()</tt></dd>
149dad308ecSroot    * <dt>DOKU_LEXER_EXIT</dt>
150dad308ecSroot    * <dd> a pattern set by <tt>addExitPattern()</tt></dd>
151dad308ecSroot    * <dt>DOKU_LEXER_SPECIAL</dt>
152dad308ecSroot    * <dd>a pattern set by <tt>addSpecialPattern()</tt></dd>
153dad308ecSroot    * <dt>DOKU_LEXER_UNMATCHED</dt>
154dad308ecSroot    * <dd>ordinary text encountered within the plugin's syntax mode
155dad308ecSroot    * which doesn't match any pattern.</dd>
156dad308ecSroot    * </dl>
157dad308ecSroot    * @param $aMatch String The text matched by the patterns.
158dad308ecSroot    * @param $aState Integer The lexer state for the match.
159dad308ecSroot    * @param $aPos Integer The character position of the matched text.
160dad308ecSroot    * @param $aHandler Object Reference to the Doku_Handler object.
161dad308ecSroot    * @return Integer The current lexer state for the match.
162dad308ecSroot    * @public
163dad308ecSroot    * @see render()
164dad308ecSroot    * @static
165dad308ecSroot    */
166dad308ecSroot    function handle($match, $state, $pos, Doku_Handler $handler)
167dad308ecSroot    {
168dad308ecSroot        switch ($state) {
169dad308ecSroot            case DOKU_LEXER_ENTER:
170dad308ecSroot                $this->_reset();        // reset object;
171dad308ecSroot
172dad308ecSroot                $opts = $this->_parseOptions(trim(substr($match,5,-1)));
173dad308ecSroot                $col = $opts['col'];
174dad308ecSroot                if (!empty($col) && is_numeric($col) && $col > 0 && $col < 5)
175dad308ecSroot                    $this->rcmd['columns'] = $col;
176dad308ecSroot                if ($opts['align'] == "left")   $this->rcmd['float'] = "left";
177dad308ecSroot                if ($opts['align'] == "center") $this->rcmd['float'] = "center";
178dad308ecSroot                if ($opts['align'] == "right")  $this->rcmd['float'] = "right";
179ddbf0766SEli Fenton                if ($opts['valign'] == "top")   $this->rcmd['valign'] = "vtop";
180ddbf0766SEli Fenton                if ($opts['valign'] == "center") $this->rcmd['valign'] = "vcenter";
181ddbf0766SEli Fenton                if ($opts['valign'] == "bottom")  $this->rcmd['valign'] = "vbottom";
182dad308ecSroot                if (!empty($opts['caption']))
183dad308ecSroot                    $this->rcmd['caption'] = hsc($opts['caption']);
184dad308ecSroot                if (!empty($opts['type']))
185dad308ecSroot                    $this->rcmd['type'] = hsc($opts['type']);
186ff0823ddSEli Fenton                if (!empty($opts['width']))
187ff0823ddSEli Fenton                    $this->rcmd['width'] = hsc($opts['width']);
188ff0823ddSEli Fenton                $this->rcmd['wrap'] = !empty($opts['wrap']);
189dad308ecSroot            break;
190dad308ecSroot          case DOKU_LEXER_MATCHED:
191dad308ecSroot
192dad308ecSroot                $menuitem = preg_split('/\|/', trim(substr($match,6,-7)));
193dad308ecSroot
194dad308ecSroot                $title = hsc($menuitem[0]);
195dad308ecSroot                if (substr($menuitem[2],0,2) == "{{")
196dad308ecSroot                    $link = $this->_itemimage($menuitem[2], $title);
197dad308ecSroot                else
198dad308ecSroot                    $link = $this->_itemLink($menuitem[2], $title);
199dad308ecSroot                $image = $this->_itemimage($menuitem[3], $title);
200dad308ecSroot
201dad308ecSroot                $this->rcmd['items'][] = array("image" => $image,
202dad308ecSroot                                               "link"  => $link,
203dad308ecSroot                                               "descr" => hsc($menuitem[1]));
204dad308ecSroot
205ff0823ddSEli Fenton                if (!empty($opts['width'])) {
206dad308ecSroot                    // find out how much space the biggest menu item needs
207dad308ecSroot                    $titlelen = mb_strlen($menuitem[0], "UTF-8");
208dad308ecSroot                    if ($titlelen > $this->rcmd['width'])
209dad308ecSroot                        $this->rcmd['width'] = $titlelen;
210ff0823ddSEli Fenton                }
211dad308ecSroot            break;
212dad308ecSroot          case DOKU_LEXER_EXIT:
213dad308ecSroot              // give the menu a convinient width. IE6 needs more space here than Firefox
214ff0823ddSEli Fenton              if (!empty($opts['width'])) {
215dad308ecSroot                  $this->rcmd['width'] += 5;
216ff0823ddSEli Fenton              }
217dad308ecSroot              return $this->rcmd;
218dad308ecSroot          default:
219dad308ecSroot            break;
220dad308ecSroot        }
221dad308ecSroot        return array();
222dad308ecSroot    }
223dad308ecSroot
224dad308ecSroot    function _reset()
225dad308ecSroot    {
226dad308ecSroot        $this->rcmd = array();
227dad308ecSroot        $this->rcmd['columns'] = 1;
228dad308ecSroot        $this->rcmd['float']   = "left";
229ddbf0766SEli Fenton        $this->rcmd['valign']  = "vtop";
230dad308ecSroot    }
231dad308ecSroot
232dad308ecSroot    function _itemlink($match, $title) {
233dad308ecSroot        // Strip the opening and closing markup
234dad308ecSroot        $link = preg_replace(array('/^\[\[/','/\]\]$/u'),'',$match);
235dad308ecSroot
236dad308ecSroot        // Split title from URL
237dad308ecSroot        $link = explode('|',$link,2);
238dad308ecSroot        $ref  = trim($link[0]);
239dad308ecSroot
240dad308ecSroot        //decide which kind of link it is
241dad308ecSroot        if ( preg_match('/^[a-zA-Z0-9\.]+>{1}.*$/u',$ref) ) {
242dad308ecSroot            // Interwiki
243dad308ecSroot            $interwiki = explode('>',$ref,2);
244dad308ecSroot            return array('interwikilink',
245dad308ecSroot                         array($ref,$title,strtolower($interwiki[0]),$interwiki[1]));
246dad308ecSroot        } elseif ( preg_match('/^\\\\\\\\[\w.:?\-;,]+?\\\\/u',$ref) ) {
247dad308ecSroot            // Windows Share
248dad308ecSroot            return array('windowssharelink', array($ref,$title));
249dad308ecSroot        } elseif ( preg_match('#^([a-z0-9\-\.+]+?)://#i',$ref) ) {
250dad308ecSroot            // external link (accepts all protocols)
251dad308ecSroot            return array('externallink', array($ref,$title));
252dad308ecSroot        } elseif ( preg_match('<'.PREG_PATTERN_VALID_EMAIL.'>',$ref) ) {
253dad308ecSroot            // E-Mail (pattern above is defined in inc/mail.php)
254dad308ecSroot            return array('emaillink', array($ref,$title));
255dad308ecSroot        } elseif ( preg_match('!^#.+!',$ref) ) {
256dad308ecSroot            // local link
257dad308ecSroot            return array('locallink', array(substr($ref,1),$title));
258dad308ecSroot        } else {
259dad308ecSroot            // internal link
260dad308ecSroot            return array('internallink', array($ref,$title));
261dad308ecSroot        }
262dad308ecSroot    }
263dad308ecSroot
264dad308ecSroot    function _itemimage($match, $title) {
265dad308ecSroot        $p = Doku_Handler_Parse_Media($match);
266dad308ecSroot
267dad308ecSroot        return array($p['type'],
268dad308ecSroot                     array($p['src'], $title, $p['align'], $p['width'],
269dad308ecSroot                     $p['height'], $p['cache'], $p['linking']));
270dad308ecSroot    }
271dad308ecSroot
272dad308ecSroot   /**
273dad308ecSroot    * Handle the actual output creation.
274dad308ecSroot    *
275dad308ecSroot    * <p>
276dad308ecSroot    * The method checks for the given <tt>$aFormat</tt> and returns
277dad308ecSroot    * <tt>FALSE</tt> when a format isn't supported. <tt>$aRenderer</tt>
278dad308ecSroot    * contains a reference to the renderer object which is currently
279dad308ecSroot    * handling the rendering. The contents of <tt>$aData</tt> is the
280dad308ecSroot    * return value of the <tt>handle()</tt> method.
281dad308ecSroot    * </p>
282dad308ecSroot    * @param $aFormat String The output format to generate.
283dad308ecSroot    * @param $aRenderer Object A reference to the renderer object.
284dad308ecSroot    * @param $aData Array The data created by the <tt>handle()</tt>
285dad308ecSroot    * method.
286dad308ecSroot    * @return Boolean <tt>TRUE</tt> if rendered successfully, or
287dad308ecSroot    * <tt>FALSE</tt> otherwise.
288dad308ecSroot    * @public
289dad308ecSroot    * @see handle()
290dad308ecSroot    */
291dad308ecSroot    function render($mode, Doku_Renderer $renderer, $data) {
292dad308ecSroot
293dad308ecSroot        if (empty($data)) return false;
294dad308ecSroot
295dad308ecSroot        if($mode == 'xhtml'){
296dad308ecSroot            if ($data['type'] != "menubar"){
297ddbf0766SEli Fenton                    $renderer->doc .= '<div class="menu menu'.$data['float'].' menu'.$data['valign'].'"';
298ff0823ddSEli Fenton                    $renderer->doc .= ' style="width:' . $data['width'] . '">'."\n";
299dad308ecSroot                    if (isset($data['caption']))
300dad308ecSroot                        $renderer->doc .= '<p class="caption">'.$data['caption'].'</p>'."\n";
301dad308ecSroot
302ff0823ddSEli Fenton                    $width = floor(100 / $data['columns']) . '%';
303ff0823ddSEli Fenton
304dad308ecSroot                    foreach($data['items'] as $item) {
305ff0823ddSEli Fenton                        $renderer->doc .= '<div class="menuitem" style="width:' . $width . '">'."\n";
306dad308ecSroot
307dad308ecSroot                        // create <img .. /> tag
308dad308ecSroot                        list($type, $args) = $item['image'];
309dad308ecSroot                        list($ext,$mime,$dl) = mimetype($args[0]);
310dad308ecSroot                        $class = ($ext == 'png') ? ' png' : NULL;
311dad308ecSroot                        $img = $renderer->_media($args[0],$args[1],$class,$args[3],$args[4],$args[5]);
312dad308ecSroot
313dad308ecSroot                        // create <a href= .. /> tag
314dad308ecSroot                        list($type, $args) = $item['link'];
315dad308ecSroot                        $link = $this->_getLink($type, $args, $renderer);
316dad308ecSroot                        $link['title'] = $args[1];
317dad308ecSroot
318dad308ecSroot                        $link['name']  = $img;
319dad308ecSroot                        $renderer->doc .= $renderer->_formatLink($link);
320dad308ecSroot
321dad308ecSroot                        $link['name']  = '<span class="menutext">'.$args[1].'</span>';
322ff0823ddSEli Fenton                        $renderer->doc .= '<div class="menutextcontainer">'."\n";
323dad308ecSroot                        $renderer->doc .= $renderer->_formatLink($link);
324dad308ecSroot                        $renderer->doc .= '<p class="menudesc">'.$item['descr'].'</p>';
325dad308ecSroot                        $renderer->doc .= '</div>'."\n";
326ff0823ddSEli Fenton
327ff0823ddSEli Fenton                        $renderer->doc .= '</div>'."\n";
328dad308ecSroot                    }
329dad308ecSroot
330dad308ecSroot                    $renderer->doc .= '</div>'."\n";
331dad308ecSroot
332ff0823ddSEli Fenton                    // Clear left/right floats, unless the 'wrap' setting is enabled.
333ff0823ddSEli Fenton                    if (!$data['wrap'] && ($data['float'] == "right" || $data['float'] == "left"))
334dad308ecSroot                        $renderer->doc .= '<p style="clear:both;" />';
335dad308ecSroot
336dad308ecSroot                    return true;
337dad308ecSroot            }
338dad308ecSroot            // menubar mode: 1 row with small captions
339dad308ecSroot            if ($data['type'] == "menubar"){
340dad308ecSroot                    $renderer->doc .= '<div id="menu"><ul class="menubar">'."\n";
341dad308ecSroot                  //  if (isset($data['caption']))
342dad308ecSroot                  //      $renderer->doc .= '<p class="caption">'.$data['caption'].'</p>'."\n";
343dad308ecSroot
344dad308ecSroot                    foreach($data['items'] as $item) {
345dad308ecSroot                        $renderer->doc .= '<li>'."\n";
346dad308ecSroot
347dad308ecSroot                        // create <img .. /> tag
348dad308ecSroot                        list($type, $args) = $item['image'];
349dad308ecSroot                        list($ext,$mime,$dl) = mimetype($args[0]);
350dad308ecSroot                        $class = ($ext == 'png') ? ' png' : NULL;
351dad308ecSroot                        $img = $renderer->_media($args[0],$item['descr'],$class,$args[3],$args[4],$args[5]);
352dad308ecSroot
353dad308ecSroot                        // create <a href= .. /> tag
354dad308ecSroot                        list($type, $args) = $item['link'];
355dad308ecSroot                        $link = $this->_getLink($type, $args, $renderer);
356dad308ecSroot                        $link['title'] = $args[1];
357dad308ecSroot
358dad308ecSroot                        $link['name']  = $img;
359dad308ecSroot                        $renderer->doc .= $renderer->_formatLink($link);
360dad308ecSroot
361dad308ecSroot                        $link['name']  = '<p class="menutext">'.$args[1].'</p>';
362dad308ecSroot                        $renderer->doc .= $renderer->_formatLink($link);
363dad308ecSroot                        //$renderer->doc .= '<p class="menudesc">'.$item['descr'].'</p>';
364dad308ecSroot                        $renderer->doc .= '</li>';
365dad308ecSroot                    }
366dad308ecSroot
367dad308ecSroot                    $renderer->doc .= '</ul></div>'."\n";
368dad308ecSroot
369dad308ecSroot                    return true;
370dad308ecSroot            }
371dad308ecSroot
372dad308ecSroot        }
373dad308ecSroot        return false;
374dad308ecSroot    }
375dad308ecSroot
376dad308ecSroot    function _createLink($url, $target=NULL)
377dad308ecSroot    {
378dad308ecSroot        global $conf;
379dad308ecSroot
380dad308ecSroot        $link = array();
381dad308ecSroot        $link['class']  = '';
382dad308ecSroot        $link['style']  = '';
383dad308ecSroot        $link['pre']    = '';
384dad308ecSroot        $link['suf']    = '';
385dad308ecSroot        $link['more']   = '';
386dad308ecSroot        $link['title']  = '';
387dad308ecSroot        $link['name']   = '';
388dad308ecSroot        $link['url']    = $url;
389dad308ecSroot
390dad308ecSroot        $link['target'] = $target == NULL ? '' : $conf['target'][$target];
391dad308ecSroot        if ($target == 'interwiki' && strpos($url,DOKU_URL) === 0) {
392dad308ecSroot            //we stay at the same server, so use local target
393dad308ecSroot            $link['target'] = $conf['target']['wiki'];
394dad308ecSroot        }
395dad308ecSroot
396dad308ecSroot        return $link;
397dad308ecSroot    }
398dad308ecSroot
399dad308ecSroot    function _getLink($type, $args, &$renderer)
400dad308ecSroot    {
401dad308ecSroot        global $ID;
402ff0823ddSEli Fenton        global $conf;
403dad308ecSroot
404dad308ecSroot        $check = false;
405dad308ecSroot        $exists = false;
406dad308ecSroot
407dad308ecSroot        switch ($type) {
408dad308ecSroot        case 'interwikilink':
409dad308ecSroot            $url  = $renderer->_resolveInterWiki($args[2],$args[3]);
410dad308ecSroot            $link = $this->_createLink($url, 'interwiki');
411dad308ecSroot            break;
412dad308ecSroot        case 'windowssharelink':
413dad308ecSroot            $url  = str_replace('\\','/',$args[0]);
414dad308ecSroot            $url  = 'file:///'.$url;
415dad308ecSroot            $link = $this->_createLink($url, 'windows');
416dad308ecSroot            break;
417dad308ecSroot        case 'externallink':
418dad308ecSroot            $link = $this->_createLink($args[0], 'extern');
419dad308ecSroot            break;
420dad308ecSroot        case 'emaillink':
421dad308ecSroot            $address = $renderer->_xmlEntities($args[0]);
422dad308ecSroot            $address = obfuscate($address);
423dad308ecSroot            if ($conf['mailguard'] == 'visible')
424dad308ecSroot                 $address = rawurlencode($address);
425dad308ecSroot
426dad308ecSroot            $link = $this->_createLink('mailto:'.$address);
427dad308ecSroot            $link['class'] = 'JSnocheck';
428dad308ecSroot            break;
429dad308ecSroot        case 'locallink':
430dad308ecSroot            $hash = sectionID($args[0], $check);
431dad308ecSroot            $link = $this->_createLink('#'.$hash);
432dad308ecSroot            $link['class'] = "wikilink1";
433dad308ecSroot            break;
434dad308ecSroot        case 'internallink':
435dad308ecSroot            resolve_pageid(getNS($ID),$args[0],$exists);
436dad308ecSroot            $url  = wl($args[0]);
437dad308ecSroot            list($id,$hash) = explode('#',$args[0],2);
438dad308ecSroot            if (!empty($hash)) $hash = sectionID($hash, $check);
439dad308ecSroot            if ($hash) $url .= '#'.$hash;    //keep hash anchor
440dad308ecSroot
441dad308ecSroot            $link = $this->_createLink($url, 'wiki');
442dad308ecSroot            $link['class'] = $exists ? 'wikilink1' : 'wikilink2';
443dad308ecSroot            break;
444dad308ecSroot        case 'internalmedia':
445dad308ecSroot            resolve_mediaid(getNS($ID),$args[0], $exists);
446dad308ecSroot            $url  = ml($args[0],array('id'=>$ID,'cache'=>$args[5]),true);
447dad308ecSroot            $link = $this->_createLink($url);
448dad308ecSroot            if (!$exists) $link['class'] = 'wikilink2';
449dad308ecSroot            break;
450dad308ecSroot        case 'externalmedia':
451dad308ecSroot            $url  = ml($args[0],array('cache'=>$args[5]));
452dad308ecSroot            $link = $this->_createLink($url);
453dad308ecSroot            break;
454dad308ecSroot        }
455dad308ecSroot        return $link;
456dad308ecSroot    }
457dad308ecSroot
458dad308ecSroot   /**
459dad308ecSroot    * Parse menu options
460dad308ecSroot    *
461dad308ecSroot    *
462dad308ecSroot    * @param $string String Option string from <menu> tag.
463dad308ecSroot    * @return array of options (name >= option). the array will be empty
464dad308ecSroot    *         if no options are found
465dad308ecSroot    * @private
466dad308ecSroot    */
467dad308ecSroot    function _parseOptions($string) {
468dad308ecSroot		$data = array();
469dad308ecSroot
470dad308ecSroot		$dq    = false;
471dad308ecSroot		$iskey = true;
472dad308ecSroot		$key   = '';
473dad308ecSroot		$val   = '';
474dad308ecSroot
475dad308ecSroot		$len = strlen($string);
476dad308ecSroot		for ($i=0; $i<=$len; $i++) {
477dad308ecSroot			// done for this one?
478dad308ecSroot			if ($string[$i] == ',' || $i == $len) {
479dad308ecSroot				$key = trim($key);
480dad308ecSroot				$val = trim($val);
481dad308ecSroot				if($key && $val) $data[strtolower($key)] = $val;
482dad308ecSroot				$iskey = true;
483dad308ecSroot				$key = '';
484dad308ecSroot				$val = '';
485dad308ecSroot				continue;
486dad308ecSroot			}
487dad308ecSroot
488dad308ecSroot			// double quotes
489dad308ecSroot			if ($string[$i] == '"') {
490dad308ecSroot				$dq = $dq ? false : true;
491dad308ecSroot				continue;
492dad308ecSroot			}
493dad308ecSroot
494dad308ecSroot			// key value separator
495dad308ecSroot			if ($string[$i] == '=' && !$dq && $iskey) {
496dad308ecSroot				$iskey = false;
497dad308ecSroot				continue;
498dad308ecSroot			}
499dad308ecSroot
500dad308ecSroot			// default
501dad308ecSroot			if ($iskey)
502dad308ecSroot				$key .= $string[$i];
503dad308ecSroot			else
504dad308ecSroot				$val .= $string[$i];
505dad308ecSroot		}
506dad308ecSroot		return $data;
507dad308ecSroot    }
508dad308ecSroot
509dad308ecSroot}
510dad308ecSroot
511dad308ecSroot//Setup VIM: ex: et ts=4 enc=utf-8 :
512dad308ecSroot?>
513