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