xref: /dokuwiki/inc/parser/xhtml.php (revision b2a412b0811b6d7fc402255e208b043ea95b79de)
10cecf9d5Sandi<?php
2b625487dSandi/**
3b625487dSandi * Renderer for XHTML output
4b625487dSandi *
5b625487dSandi * @author Harry Fuecks <hfuecks@gmail.com>
6b625487dSandi * @author Andreas Gohr <andi@splitbrain.org>
7b625487dSandi */
8b625487dSandi
90cecf9d5Sandiif(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../../').'/');
100cecf9d5Sandi
110cecf9d5Sandiif ( !defined('DOKU_LF') ) {
120cecf9d5Sandi    // Some whitespace to help View > Source
130cecf9d5Sandi    define ('DOKU_LF',"\n");
140cecf9d5Sandi}
150cecf9d5Sandi
160cecf9d5Sandiif ( !defined('DOKU_TAB') ) {
170cecf9d5Sandi    // Some whitespace to help View > Source
180cecf9d5Sandi    define ('DOKU_TAB',"\t");
190cecf9d5Sandi}
200cecf9d5Sandi
210e1c636eSandirequire_once DOKU_INC . 'inc/parser/renderer.php';
2256fa9a3fSAndreas Gohrrequire_once DOKU_INC . 'inc/html.php';
230e1c636eSandi
240cecf9d5Sandi/**
25b625487dSandi * The Renderer
260cecf9d5Sandi */
27ac83b9d8Sandiclass Doku_Renderer_xhtml extends Doku_Renderer {
280cecf9d5Sandi
29c5a8fd96SAndreas Gohr    // @access public
30c5a8fd96SAndreas Gohr    var $doc = '';        // will contain the whole document
31c5a8fd96SAndreas Gohr    var $toc = array();   // will contain the Table of Contents
32c5a8fd96SAndreas Gohr
330cecf9d5Sandi
340cecf9d5Sandi    var $headers = array();
350cecf9d5Sandi
360cecf9d5Sandi    var $footnotes = array();
370cecf9d5Sandi
380cecf9d5Sandi    var $acronyms = array();
390cecf9d5Sandi    var $smileys = array();
400cecf9d5Sandi    var $badwords = array();
410cecf9d5Sandi    var $entities = array();
420cecf9d5Sandi    var $interwiki = array();
430cecf9d5Sandi
44af587fa8Sandi    var $lastsec = 0;
45af587fa8Sandi
467764a90aSandi    var $store = '';
477764a90aSandi
480cecf9d5Sandi    function document_start() {
49c5a8fd96SAndreas Gohr        //reset some internals
50c5a8fd96SAndreas Gohr        $this->toc     = array();
51c5a8fd96SAndreas Gohr        $this->headers = array();
520cecf9d5Sandi    }
530cecf9d5Sandi
540cecf9d5Sandi    function document_end() {
550cecf9d5Sandi        if ( count ($this->footnotes) > 0 ) {
56a2d649c4Sandi            $this->doc .= '<div class="footnotes">'.DOKU_LF;
57d74aace9Schris
58d74aace9Schris            $id = 0;
590cecf9d5Sandi            foreach ( $this->footnotes as $footnote ) {
60d74aace9Schris                $id++;   // the number of the current footnote
61d74aace9Schris
62d74aace9Schris                // check its not a placeholder that indicates actual footnote text is elsewhere
63d74aace9Schris                if (substr($footnote, 0, 5) != "@@FNT") {
64d74aace9Schris
65d74aace9Schris                    // open the footnote and set the anchor and backlink
66d74aace9Schris                    $this->doc .= '<div class="fn">';
6724a33b42SAndreas Gohr                    $this->doc .= '<a href="#fnt__'.$id.'" id="fn__'.$id.'" name="fn__'.$id.'" class="fn_bot">';
68d74aace9Schris                    $this->doc .= $id.')</a> '.DOKU_LF;
69d74aace9Schris
70d74aace9Schris                    // get any other footnotes that use the same markup
71d74aace9Schris                    $alt = array_keys($this->footnotes, "@@FNT$id");
72d74aace9Schris
73d74aace9Schris                    if (count($alt)) {
74d74aace9Schris                      foreach ($alt as $ref) {
75d74aace9Schris                        // set anchor and backlink for the other footnotes
7624a33b42SAndreas Gohr                        $this->doc .= ', <a href="#fnt__'.($ref+1).'" id="fn__'.($ref+1).'" name="fn__'.($ref+1).'" class="fn_bot">';
77d74aace9Schris                        $this->doc .= ($ref+1).')</a> '.DOKU_LF;
78d74aace9Schris                      }
79d74aace9Schris                    }
80d74aace9Schris
81d74aace9Schris                    // add footnote markup and close this footnote
82a2d649c4Sandi                    $this->doc .= $footnote;
83d74aace9Schris                    $this->doc .= '</div>' . DOKU_LF;
84d74aace9Schris                }
850cecf9d5Sandi            }
86a2d649c4Sandi            $this->doc .= '</div>'.DOKU_LF;
870cecf9d5Sandi        }
88c5a8fd96SAndreas Gohr
89c5a8fd96SAndreas Gohr        // prepend the TOC
90e41c4da9SAndreas Gohr        if($this->info['toc']){
91c5a8fd96SAndreas Gohr            $this->doc = $this->render_TOC().$this->doc;
920cecf9d5Sandi        }
933e55d035SAndreas Gohr
943e55d035SAndreas Gohr        // make sure there are no empty paragraphs
9527918226Schris        $this->doc = preg_replace('#<p>\s*</p>#','',$this->doc);
96e41c4da9SAndreas Gohr    }
970cecf9d5Sandi
98c5a8fd96SAndreas Gohr    /**
99c5a8fd96SAndreas Gohr     * Return the TOC rendered to XHTML
100c5a8fd96SAndreas Gohr     *
101c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
102c5a8fd96SAndreas Gohr     */
103c5a8fd96SAndreas Gohr    function render_TOC(){
104e41c4da9SAndreas Gohr        if(count($this->toc) < 3) return '';
105c5a8fd96SAndreas Gohr        global $lang;
106c5a8fd96SAndreas Gohr        $out  = '<div class="toc">'.DOKU_LF;
107c5a8fd96SAndreas Gohr        $out .= '<div class="tocheader toctoggle" id="toc__header">';
108c5a8fd96SAndreas Gohr        $out .= $lang['toc'];
109c5a8fd96SAndreas Gohr        $out .= '</div>'.DOKU_LF;
110c5a8fd96SAndreas Gohr        $out .= '<div id="toc__inside">'.DOKU_LF;
111c5a8fd96SAndreas Gohr        $out .= html_buildlist($this->toc,'toc',array($this,'_tocitem'));
112c5a8fd96SAndreas Gohr        $out .= '</div>'.DOKU_LF.'</div>'.DOKU_LF;
113c5a8fd96SAndreas Gohr        return $out;
114c5a8fd96SAndreas Gohr    }
115c5a8fd96SAndreas Gohr
116c5a8fd96SAndreas Gohr    /**
117c5a8fd96SAndreas Gohr     * Callback for html_buildlist
118c5a8fd96SAndreas Gohr     */
119c5a8fd96SAndreas Gohr    function _tocitem($item){
120c5a8fd96SAndreas Gohr        return '<span class="li"><a href="#'.$item['hid'].'" class="toc">'.
121c5a8fd96SAndreas Gohr               $this->_xmlEntities($item['title']).'</a></span>';
122c5a8fd96SAndreas Gohr    }
123c5a8fd96SAndreas Gohr
124af587fa8Sandi    function header($text, $level, $pos) {
125af587fa8Sandi        global $conf;
126af587fa8Sandi
127c5a8fd96SAndreas Gohr        // create a unique header id
128c5a8fd96SAndreas Gohr        $hid = $this->_headerToLink($text,'true');
129c5a8fd96SAndreas Gohr
130c5a8fd96SAndreas Gohr        //handle TOC
131c5a8fd96SAndreas Gohr        if($level >= $conf['toptoclevel'] && $level <= $conf['maxtoclevel']){
132c5a8fd96SAndreas Gohr            // the TOC is one of our standard ul list arrays ;-)
133c5a8fd96SAndreas Gohr            $this->toc[] = array( 'hid'   => $hid,
134c5a8fd96SAndreas Gohr                                  'title' => $text,
135c5a8fd96SAndreas Gohr                                  'type'  => 'ul',
13621ed2ba4Sjan                                  'level' => $level-$conf['toptoclevel']+1);
137c5a8fd96SAndreas Gohr        }
138c5a8fd96SAndreas Gohr
139c5a8fd96SAndreas Gohr        // write the header
140c5a8fd96SAndreas Gohr        $this->doc .= DOKU_LF.'<h'.$level.'><a name="'.$hid.'" id="'.$hid.'">';
141a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
142c6a1c1bfSAnika Henke        $this->doc .= "</a></h$level>".DOKU_LF;
1430cecf9d5Sandi    }
1440cecf9d5Sandi
14535dae8b0SBen Coburn     /**
14635dae8b0SBen Coburn     * Section edit marker is replaced by an edit button when
14735dae8b0SBen Coburn     * the page is editable. Replacement done in 'inc/html.php#html_secedit'
14835dae8b0SBen Coburn     *
14935dae8b0SBen Coburn     * @author Andreas Gohr <andi@splitbrain.org>
15035dae8b0SBen Coburn     * @author Ben Coburn   <btcoburn@silicodon.net>
15135dae8b0SBen Coburn     */
15235dae8b0SBen Coburn    function section_edit($start, $end, $level, $name) {
15335dae8b0SBen Coburn        global $conf;
15435dae8b0SBen Coburn
15535dae8b0SBen Coburn        if ($start!=-1 && $level<=$conf['maxseclevel']) {
15635dae8b0SBen Coburn            $name = str_replace('"', '', $name);
15735dae8b0SBen Coburn            $this->doc .= '<!-- SECTION "'.$name.'" ['.$start.'-'.(($end===0)?'':$end).'] -->';
15835dae8b0SBen Coburn        }
15935dae8b0SBen Coburn    }
16035dae8b0SBen Coburn
1610cecf9d5Sandi    function section_open($level) {
162a2d649c4Sandi        $this->doc .= "<div class=\"level$level\">".DOKU_LF;
1630cecf9d5Sandi    }
1640cecf9d5Sandi
1650cecf9d5Sandi    function section_close() {
166a2d649c4Sandi        $this->doc .= DOKU_LF.'</div>'.DOKU_LF;
1670cecf9d5Sandi    }
1680cecf9d5Sandi
1690cecf9d5Sandi    function cdata($text) {
170a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
1710cecf9d5Sandi    }
1720cecf9d5Sandi
1730cecf9d5Sandi    function p_open() {
174a2d649c4Sandi        $this->doc .= DOKU_LF.'<p>'.DOKU_LF;
1750cecf9d5Sandi    }
1760cecf9d5Sandi
1770cecf9d5Sandi    function p_close() {
178a2d649c4Sandi        $this->doc .= DOKU_LF.'</p>'.DOKU_LF;
1790cecf9d5Sandi    }
1800cecf9d5Sandi
1810cecf9d5Sandi    function linebreak() {
182a2d649c4Sandi        $this->doc .= '<br/>'.DOKU_LF;
1830cecf9d5Sandi    }
1840cecf9d5Sandi
1850cecf9d5Sandi    function hr() {
1864beabca9SAnika Henke        $this->doc .= '<hr />'.DOKU_LF;
1870cecf9d5Sandi    }
1880cecf9d5Sandi
1890cecf9d5Sandi    function strong_open() {
190a2d649c4Sandi        $this->doc .= '<strong>';
1910cecf9d5Sandi    }
1920cecf9d5Sandi
1930cecf9d5Sandi    function strong_close() {
194a2d649c4Sandi        $this->doc .= '</strong>';
1950cecf9d5Sandi    }
1960cecf9d5Sandi
1970cecf9d5Sandi    function emphasis_open() {
198a2d649c4Sandi        $this->doc .= '<em>';
1990cecf9d5Sandi    }
2000cecf9d5Sandi
2010cecf9d5Sandi    function emphasis_close() {
202a2d649c4Sandi        $this->doc .= '</em>';
2030cecf9d5Sandi    }
2040cecf9d5Sandi
2050cecf9d5Sandi    function underline_open() {
20602e51121SAnika Henke        $this->doc .= '<em class="u">';
2070cecf9d5Sandi    }
2080cecf9d5Sandi
2090cecf9d5Sandi    function underline_close() {
21002e51121SAnika Henke        $this->doc .= '</em>';
2110cecf9d5Sandi    }
2120cecf9d5Sandi
2130cecf9d5Sandi    function monospace_open() {
214a2d649c4Sandi        $this->doc .= '<code>';
2150cecf9d5Sandi    }
2160cecf9d5Sandi
2170cecf9d5Sandi    function monospace_close() {
218a2d649c4Sandi        $this->doc .= '</code>';
2190cecf9d5Sandi    }
2200cecf9d5Sandi
2210cecf9d5Sandi    function subscript_open() {
222a2d649c4Sandi        $this->doc .= '<sub>';
2230cecf9d5Sandi    }
2240cecf9d5Sandi
2250cecf9d5Sandi    function subscript_close() {
226a2d649c4Sandi        $this->doc .= '</sub>';
2270cecf9d5Sandi    }
2280cecf9d5Sandi
2290cecf9d5Sandi    function superscript_open() {
230a2d649c4Sandi        $this->doc .= '<sup>';
2310cecf9d5Sandi    }
2320cecf9d5Sandi
2330cecf9d5Sandi    function superscript_close() {
234a2d649c4Sandi        $this->doc .= '</sup>';
2350cecf9d5Sandi    }
2360cecf9d5Sandi
2370cecf9d5Sandi    function deleted_open() {
238a2d649c4Sandi        $this->doc .= '<del>';
2390cecf9d5Sandi    }
2400cecf9d5Sandi
2410cecf9d5Sandi    function deleted_close() {
242a2d649c4Sandi        $this->doc .= '</del>';
2430cecf9d5Sandi    }
2440cecf9d5Sandi
2453fd0b676Sandi    /**
2463fd0b676Sandi     * Callback for footnote start syntax
2473fd0b676Sandi     *
2483fd0b676Sandi     * All following content will go to the footnote instead of
249d74aace9Schris     * the document. To achieve this the previous rendered content
2503fd0b676Sandi     * is moved to $store and $doc is cleared
2513fd0b676Sandi     *
2523fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
2533fd0b676Sandi     */
2540cecf9d5Sandi    function footnote_open() {
2557764a90aSandi
2567764a90aSandi        // move current content to store and record footnote
2577764a90aSandi        $this->store = $this->doc;
2587764a90aSandi        $this->doc   = '';
2590cecf9d5Sandi    }
2600cecf9d5Sandi
2613fd0b676Sandi    /**
2623fd0b676Sandi     * Callback for footnote end syntax
2633fd0b676Sandi     *
2643fd0b676Sandi     * All rendered content is moved to the $footnotes array and the old
2653fd0b676Sandi     * content is restored from $store again
2663fd0b676Sandi     *
2673fd0b676Sandi     * @author Andreas Gohr
2683fd0b676Sandi     */
2690cecf9d5Sandi    function footnote_close() {
2707764a90aSandi
271d74aace9Schris        // recover footnote into the stack and restore old content
272d74aace9Schris        $footnote = $this->doc;
2737764a90aSandi        $this->doc = $this->store;
2747764a90aSandi        $this->store = '';
275d74aace9Schris
276d74aace9Schris        // check to see if this footnote has been seen before
277d74aace9Schris        $i = array_search($footnote, $this->footnotes);
278d74aace9Schris
279d74aace9Schris        if ($i === false) {
280d74aace9Schris            // its a new footnote, add it to the $footnotes array
281d74aace9Schris            $id = count($this->footnotes)+1;
282d74aace9Schris            $this->footnotes[count($this->footnotes)] = $footnote;
283d74aace9Schris        } else {
284d74aace9Schris            // seen this one before, translate the index to an id and save a placeholder
285d74aace9Schris            $i++;
286d74aace9Schris            $id = count($this->footnotes)+1;
287d74aace9Schris            $this->footnotes[count($this->footnotes)] = "@@FNT".($i);
288d74aace9Schris        }
289d74aace9Schris
290d74aace9Schris        // output the footnote reference and link, incl. onmouseover for insitu footnote popup
29124a33b42SAndreas Gohr        $this->doc .= '<a href="#fn__'.$id.'" name="fnt__'.$id.'" id="fnt__'.$id.'" class="fn_top" onmouseover="fnt(\''.$id.'\', this, event);">'.$id.')</a>';
2920cecf9d5Sandi    }
2930cecf9d5Sandi
2940cecf9d5Sandi    function listu_open() {
295a2d649c4Sandi        $this->doc .= '<ul>'.DOKU_LF;
2960cecf9d5Sandi    }
2970cecf9d5Sandi
2980cecf9d5Sandi    function listu_close() {
299a2d649c4Sandi        $this->doc .= '</ul>'.DOKU_LF;
3000cecf9d5Sandi    }
3010cecf9d5Sandi
3020cecf9d5Sandi    function listo_open() {
303a2d649c4Sandi        $this->doc .= '<ol>'.DOKU_LF;
3040cecf9d5Sandi    }
3050cecf9d5Sandi
3060cecf9d5Sandi    function listo_close() {
307a2d649c4Sandi        $this->doc .= '</ol>'.DOKU_LF;
3080cecf9d5Sandi    }
3090cecf9d5Sandi
3100cecf9d5Sandi    function listitem_open($level) {
311a2d649c4Sandi        $this->doc .= '<li class="level'.$level.'">';
3120cecf9d5Sandi    }
3130cecf9d5Sandi
3140cecf9d5Sandi    function listitem_close() {
315a2d649c4Sandi        $this->doc .= '</li>'.DOKU_LF;
3160cecf9d5Sandi    }
3170cecf9d5Sandi
3180cecf9d5Sandi    function listcontent_open() {
31990db23d7Schris        $this->doc .= '<div class="li">';
3200cecf9d5Sandi    }
3210cecf9d5Sandi
3220cecf9d5Sandi    function listcontent_close() {
32390db23d7Schris        $this->doc .= '</div>'.DOKU_LF;
3240cecf9d5Sandi    }
3250cecf9d5Sandi
3260cecf9d5Sandi    function unformatted($text) {
327a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
3280cecf9d5Sandi    }
3290cecf9d5Sandi
3300cecf9d5Sandi    /**
3313fd0b676Sandi     * Execute PHP code if allowed
3323fd0b676Sandi     *
3333fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3340cecf9d5Sandi     */
3350cecf9d5Sandi    function php($text) {
3364de671bcSandi        global $conf;
3374de671bcSandi        if($conf['phpok']){
338bad0b545Sandi            ob_start();
3394de671bcSandi            eval($text);
3403fd0b676Sandi            $this->doc .= ob_get_contents();
341bad0b545Sandi            ob_end_clean();
3424de671bcSandi        }else{
3434de671bcSandi            $this->file($text);
3444de671bcSandi        }
3450cecf9d5Sandi    }
3460cecf9d5Sandi
3470cecf9d5Sandi    /**
3483fd0b676Sandi     * Insert HTML if allowed
3493fd0b676Sandi     *
3503fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3510cecf9d5Sandi     */
3520cecf9d5Sandi    function html($text) {
3534de671bcSandi        global $conf;
3544de671bcSandi        if($conf['htmlok']){
355a2d649c4Sandi          $this->doc .= $text;
3564de671bcSandi        }else{
3570cecf9d5Sandi          $this->file($text);
3580cecf9d5Sandi        }
3594de671bcSandi    }
3600cecf9d5Sandi
3610cecf9d5Sandi    function preformatted($text) {
362a2d649c4Sandi        $this->doc .= '<pre class="code">' . $this->_xmlEntities($text) . '</pre>'. DOKU_LF;
3630cecf9d5Sandi    }
3640cecf9d5Sandi
3650cecf9d5Sandi    function file($text) {
366a2d649c4Sandi        $this->doc .= '<pre class="file">' . $this->_xmlEntities($text). '</pre>'. DOKU_LF;
3670cecf9d5Sandi    }
3680cecf9d5Sandi
3690cecf9d5Sandi    function quote_open() {
37096331712SAnika Henke        $this->doc .= '<blockquote><div class="no">'.DOKU_LF;
3710cecf9d5Sandi    }
3720cecf9d5Sandi
3730cecf9d5Sandi    function quote_close() {
37496331712SAnika Henke        $this->doc .= '</div></blockquote>'.DOKU_LF;
3750cecf9d5Sandi    }
3760cecf9d5Sandi
3770cecf9d5Sandi    /**
3783fd0b676Sandi     * Callback for code text
3793fd0b676Sandi     *
3803fd0b676Sandi     * Uses GeSHi to highlight language syntax
3813fd0b676Sandi     *
3823fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3830cecf9d5Sandi     */
3840cecf9d5Sandi    function code($text, $language = NULL) {
3854de671bcSandi        global $conf;
3860cecf9d5Sandi
3870cecf9d5Sandi        if ( is_null($language) ) {
3880cecf9d5Sandi            $this->preformatted($text);
3890cecf9d5Sandi        } else {
390852896daSAndreas Gohr            //strip leading and trailing blank line
391313da78aSandi            $text = preg_replace('/^\s*?\n/','',$text);
392852896daSAndreas Gohr            $text = preg_replace('/\s*?\n$/','',$text);
3938f7d700cSchris            $this->doc .= p_xhtml_cached_geshi($text, $language);
3940cecf9d5Sandi        }
3950cecf9d5Sandi    }
3960cecf9d5Sandi
3970cecf9d5Sandi    function acronym($acronym) {
3980cecf9d5Sandi
3990cecf9d5Sandi        if ( array_key_exists($acronym, $this->acronyms) ) {
4000cecf9d5Sandi
401433bef32Sandi            $title = $this->_xmlEntities($this->acronyms[$acronym]);
4020cecf9d5Sandi
403a2d649c4Sandi            $this->doc .= '<acronym title="'.$title
404433bef32Sandi                .'">'.$this->_xmlEntities($acronym).'</acronym>';
4050cecf9d5Sandi
4060cecf9d5Sandi        } else {
407a2d649c4Sandi            $this->doc .= $this->_xmlEntities($acronym);
4080cecf9d5Sandi        }
4090cecf9d5Sandi    }
4100cecf9d5Sandi
4110cecf9d5Sandi    function smiley($smiley) {
4120cecf9d5Sandi        if ( array_key_exists($smiley, $this->smileys) ) {
413433bef32Sandi            $title = $this->_xmlEntities($this->smileys[$smiley]);
414f62ea8a1Sandi            $this->doc .= '<img src="'.DOKU_BASE.'lib/images/smileys/'.$this->smileys[$smiley].
4154beabca9SAnika Henke                '" class="middle" alt="'.
416433bef32Sandi                    $this->_xmlEntities($smiley).'" />';
4170cecf9d5Sandi        } else {
418a2d649c4Sandi            $this->doc .= $this->_xmlEntities($smiley);
4190cecf9d5Sandi        }
4200cecf9d5Sandi    }
4210cecf9d5Sandi
422f62ea8a1Sandi    /*
4234de671bcSandi    * not used
4240cecf9d5Sandi    function wordblock($word) {
4250cecf9d5Sandi        if ( array_key_exists($word, $this->badwords) ) {
426a2d649c4Sandi            $this->doc .= '** BLEEP **';
4270cecf9d5Sandi        } else {
428a2d649c4Sandi            $this->doc .= $this->_xmlEntities($word);
4290cecf9d5Sandi        }
4300cecf9d5Sandi    }
4314de671bcSandi    */
4320cecf9d5Sandi
4330cecf9d5Sandi    function entity($entity) {
4340cecf9d5Sandi        if ( array_key_exists($entity, $this->entities) ) {
435a2d649c4Sandi            $this->doc .= $this->entities[$entity];
4360cecf9d5Sandi        } else {
437a2d649c4Sandi            $this->doc .= $this->_xmlEntities($entity);
4380cecf9d5Sandi        }
4390cecf9d5Sandi    }
4400cecf9d5Sandi
4410cecf9d5Sandi    function multiplyentity($x, $y) {
442a2d649c4Sandi        $this->doc .= "$x&times;$y";
4430cecf9d5Sandi    }
4440cecf9d5Sandi
4450cecf9d5Sandi    function singlequoteopening() {
446a2d649c4Sandi        $this->doc .= "&lsquo;";
4470cecf9d5Sandi    }
4480cecf9d5Sandi
4490cecf9d5Sandi    function singlequoteclosing() {
450a2d649c4Sandi        $this->doc .= "&rsquo;";
4510cecf9d5Sandi    }
4520cecf9d5Sandi
4530cecf9d5Sandi    function doublequoteopening() {
454a2d649c4Sandi        $this->doc .= "&ldquo;";
4550cecf9d5Sandi    }
4560cecf9d5Sandi
4570cecf9d5Sandi    function doublequoteclosing() {
458a2d649c4Sandi        $this->doc .= "&rdquo;";
4590cecf9d5Sandi    }
4600cecf9d5Sandi
4610cecf9d5Sandi    /**
4620cecf9d5Sandi    */
4630cecf9d5Sandi    function camelcaselink($link) {
46411d0aa47Sandi      $this->internallink($link,$link);
4650cecf9d5Sandi    }
4660cecf9d5Sandi
4670b7c14c2Sandi
4680b7c14c2Sandi    function locallink($hash, $name = NULL){
4690b7c14c2Sandi        global $ID;
4700b7c14c2Sandi        $name  = $this->_getLinkTitle($name, $hash, $isImage);
4710b7c14c2Sandi        $hash  = $this->_headerToLink($hash);
4720b7c14c2Sandi        $title = $ID.' &crarr;';
4730b7c14c2Sandi        $this->doc .= '<a href="#'.$hash.'" title="'.$title.'" class="wikilink1">';
4740b7c14c2Sandi        $this->doc .= $name;
4750b7c14c2Sandi        $this->doc .= '</a>';
4760b7c14c2Sandi    }
4770b7c14c2Sandi
478cffcc403Sandi    /**
4793fd0b676Sandi     * Render an internal Wiki Link
4803fd0b676Sandi     *
481cffcc403Sandi     * $search and $returnonly are not for the renderer but are used
482cffcc403Sandi     * elsewhere - no need to implement them in other renderers
4833fd0b676Sandi     *
4843fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
485cffcc403Sandi     */
486cffcc403Sandi    function internallink($id, $name = NULL, $search=NULL,$returnonly=false) {
487ba11bd29Sandi        global $conf;
48837e34a5eSandi        global $ID;
4890339c872Sjan        // default name is based on $id as given
4900339c872Sjan        $default = $this->_simpleTitle($id);
4910339c872Sjan        // now first resolve and clean up the $id
49237e34a5eSandi        resolve_pageid(getNS($ID),$id,$exists);
4930339c872Sjan        $name = $this->_getLinkTitle($name, $default, $isImage, $id);
4940e1c636eSandi        if ( !$isImage ) {
4950e1c636eSandi            if ( $exists ) {
496ba11bd29Sandi                $class='wikilink1';
4970cecf9d5Sandi            } else {
498ba11bd29Sandi                $class='wikilink2';
4990cecf9d5Sandi            }
5000cecf9d5Sandi        } else {
501ba11bd29Sandi            $class='media';
5020cecf9d5Sandi        }
5030cecf9d5Sandi
504a1685bedSandi        //keep hash anchor
505a1685bedSandi        list($id,$hash) = split('#',$id,2);
506a1685bedSandi
507ba11bd29Sandi        //prepare for formating
508ba11bd29Sandi        $link['target'] = $conf['target']['wiki'];
509ba11bd29Sandi        $link['style']  = '';
510ba11bd29Sandi        $link['pre']    = '';
511ba11bd29Sandi        $link['suf']    = '';
51240eb54bbSjan        // highlight link to current page
51340eb54bbSjan        if ($id == $ID) {
51492795d04Sandi            $link['pre']    = '<span class="curid">';
51592795d04Sandi            $link['suf']    = '</span>';
51640eb54bbSjan        }
5175e163278SAndreas Gohr        $link['more']   = '';
518ba11bd29Sandi        $link['class']  = $class;
519ba11bd29Sandi        $link['url']    = wl($id);
520ba11bd29Sandi        $link['name']   = $name;
521ba11bd29Sandi        $link['title']  = $id;
522723d78dbSandi        //add search string
523723d78dbSandi        if($search){
524723d78dbSandi            ($conf['userewrite']) ? $link['url'].='?s=' : $link['url'].='&amp;s=';
525b6c6979fSAndreas Gohr            $link['url'] .= rawurlencode($search);
526723d78dbSandi        }
527723d78dbSandi
528a1685bedSandi        //keep hash
529a1685bedSandi        if($hash) $link['url'].='#'.$hash;
530a1685bedSandi
531ba11bd29Sandi        //output formatted
532cffcc403Sandi        if($returnonly){
533cffcc403Sandi            return $this->_formatLink($link);
534cffcc403Sandi        }else{
535a2d649c4Sandi            $this->doc .= $this->_formatLink($link);
5360cecf9d5Sandi        }
537cffcc403Sandi    }
5380cecf9d5Sandi
539b625487dSandi    function externallink($url, $name = NULL) {
540b625487dSandi        global $conf;
5410cecf9d5Sandi
542433bef32Sandi        $name = $this->_getLinkTitle($name, $url, $isImage);
5436f0c5dbfSandi
5446f0c5dbfSandi        // add protocol on simple short URLs
54526b8b699Sandi        if(substr($url,0,3) == 'ftp' && (substr($url,0,6) != 'ftp://')) $url = 'ftp://'.$url;
5466f0c5dbfSandi        if(substr($url,0,3) == 'www') $url = 'http://'.$url;
5470cecf9d5Sandi
5480cecf9d5Sandi        if ( !$isImage ) {
549b625487dSandi            $class='urlextern';
5500cecf9d5Sandi        } else {
551b625487dSandi            $class='media';
5520cecf9d5Sandi        }
5530cecf9d5Sandi
554b625487dSandi        //prepare for formating
555b625487dSandi        $link['target'] = $conf['target']['extern'];
556b625487dSandi        $link['style']  = '';
557b625487dSandi        $link['pre']    = '';
558b625487dSandi        $link['suf']    = '';
5595e163278SAndreas Gohr        $link['more']   = '';
560b625487dSandi        $link['class']  = $class;
561b625487dSandi        $link['url']    = $url;
562e1c10e4dSchris
563b625487dSandi        $link['name']   = $name;
564433bef32Sandi        $link['title']  = $this->_xmlEntities($url);
565b625487dSandi        if($conf['relnofollow']) $link['more'] .= ' rel="nofollow"';
5660cecf9d5Sandi
567b625487dSandi        //output formatted
568a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
5690cecf9d5Sandi    }
5700cecf9d5Sandi
5710cecf9d5Sandi    /**
5720cecf9d5Sandi    */
57397a3e4e3Sandi    function interwikilink($match, $name = NULL, $wikiName, $wikiUri) {
574b625487dSandi        global $conf;
5750cecf9d5Sandi
57697a3e4e3Sandi        $link = array();
57797a3e4e3Sandi        $link['target'] = $conf['target']['interwiki'];
57897a3e4e3Sandi        $link['pre']    = '';
57997a3e4e3Sandi        $link['suf']    = '';
5805e163278SAndreas Gohr        $link['more']   = '';
581433bef32Sandi        $link['name']   = $this->_getLinkTitle($name, $wikiUri, $isImage);
5820cecf9d5Sandi
58397a3e4e3Sandi        //get interwiki URL
58497a3e4e3Sandi        if ( isset($this->interwiki[$wikiName]) ) {
58597a3e4e3Sandi            $url = $this->interwiki[$wikiName];
58697a3e4e3Sandi        } else {
58797a3e4e3Sandi            // Default to Google I'm feeling lucky
58897a3e4e3Sandi            $url = 'http://www.google.com/search?q={URL}&amp;btnI=lucky';
58997a3e4e3Sandi            $wikiName = 'go';
5900cecf9d5Sandi        }
5910cecf9d5Sandi
59297a3e4e3Sandi        if ( !$isImage ) {
5939d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$wikiName);
5949d2ddea4SAndreas Gohr            $link['class'] = "interwiki iw_$class";
5951c2d1019SAndreas Gohr        } else {
5961c2d1019SAndreas Gohr            $link['class'] = 'media';
59797a3e4e3Sandi        }
5980cecf9d5Sandi
59997a3e4e3Sandi        //do we stay at the same server? Use local target
60097a3e4e3Sandi        if( strpos($url,DOKU_URL) === 0 ){
60197a3e4e3Sandi            $link['target'] = $conf['target']['wiki'];
60297a3e4e3Sandi        }
6030cecf9d5Sandi
604b034216dSAndreas Gohr        //split into hash and url part
605b034216dSAndreas Gohr        list($wikiUri,$hash) = explode('#',$wikiUri,2);
606b034216dSAndreas Gohr
60797a3e4e3Sandi        //replace placeholder
60897a3e4e3Sandi        if(preg_match('#\{(URL|NAME|SCHEME|HOST|PORT|PATH|QUERY)\}#',$url)){
60997a3e4e3Sandi            //use placeholders
610b6c6979fSAndreas Gohr            $url = str_replace('{URL}',rawurlencode($wikiUri),$url);
61197a3e4e3Sandi            $url = str_replace('{NAME}',$wikiUri,$url);
61297a3e4e3Sandi            $parsed = parse_url($wikiUri);
61397a3e4e3Sandi            if(!$parsed['port']) $parsed['port'] = 80;
61497a3e4e3Sandi            $url = str_replace('{SCHEME}',$parsed['scheme'],$url);
61597a3e4e3Sandi            $url = str_replace('{HOST}',$parsed['host'],$url);
61697a3e4e3Sandi            $url = str_replace('{PORT}',$parsed['port'],$url);
61797a3e4e3Sandi            $url = str_replace('{PATH}',$parsed['path'],$url);
61897a3e4e3Sandi            $url = str_replace('{QUERY}',$parsed['query'],$url);
61997a3e4e3Sandi            $link['url'] = $url;
62097a3e4e3Sandi        }else{
62197a3e4e3Sandi            //default
622b6c6979fSAndreas Gohr            $link['url'] = $url.rawurlencode($wikiUri);
62397a3e4e3Sandi        }
624b6c6979fSAndreas Gohr        if($hash) $link['url'] .= '#'.rawurlencode($hash);
62597a3e4e3Sandi
62697a3e4e3Sandi        $link['title'] = htmlspecialchars($link['url']);
62797a3e4e3Sandi
62897a3e4e3Sandi        //output formatted
629a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6300cecf9d5Sandi    }
6310cecf9d5Sandi
6320cecf9d5Sandi    /**
6330cecf9d5Sandi     */
6341d47afe1Sandi    function windowssharelink($url, $name = NULL) {
6351d47afe1Sandi        global $conf;
6361d47afe1Sandi        global $lang;
6371d47afe1Sandi        //simple setup
6381d47afe1Sandi        $link['target'] = $conf['target']['windows'];
6391d47afe1Sandi        $link['pre']    = '';
6401d47afe1Sandi        $link['suf']   = '';
6411d47afe1Sandi        $link['style']  = '';
6421d47afe1Sandi        //Display error on browsers other than IE
6431d47afe1Sandi        $link['more'] = 'onclick="if(document.all == null){alert(\''.
644433bef32Sandi                        $this->_xmlEntities($lang['nosmblinks'],ENT_QUOTES).
6451d47afe1Sandi                        '\');}" onkeypress="if(document.all == null){alert(\''.
646433bef32Sandi                        $this->_xmlEntities($lang['nosmblinks'],ENT_QUOTES).'\');}"';
6470cecf9d5Sandi
648433bef32Sandi        $link['name'] = $this->_getLinkTitle($name, $url, $isImage);
6490cecf9d5Sandi        if ( !$isImage ) {
6501d47afe1Sandi            $link['class'] = 'windows';
6510cecf9d5Sandi        } else {
6521d47afe1Sandi            $link['class'] = 'media';
6530cecf9d5Sandi        }
6540cecf9d5Sandi
6550cecf9d5Sandi
656433bef32Sandi        $link['title'] = $this->_xmlEntities($url);
6571d47afe1Sandi        $url = str_replace('\\','/',$url);
6581d47afe1Sandi        $url = 'file:///'.$url;
6591d47afe1Sandi        $link['url'] = $url;
6600cecf9d5Sandi
6611d47afe1Sandi        //output formatted
662a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6630cecf9d5Sandi    }
6640cecf9d5Sandi
66571352defSandi    function emaillink($address, $name = NULL) {
66671352defSandi        global $conf;
66771352defSandi        //simple setup
66871352defSandi        $link = array();
66971352defSandi        $link['target'] = '';
67071352defSandi        $link['pre']    = '';
67171352defSandi        $link['suf']   = '';
67271352defSandi        $link['style']  = '';
67371352defSandi        $link['more']   = '';
6740cecf9d5Sandi
67571352defSandi        //we just test for image here - we need to encode the title our self
676433bef32Sandi        $this->_getLinkTitle($name, $address, $isImage);
6770cecf9d5Sandi        if ( !$isImage ) {
678776b36ecSAndreas Gohr            $link['class']='mail JSnocheck';
6790cecf9d5Sandi        } else {
680776b36ecSAndreas Gohr            $link['class']='media JSnocheck';
6810cecf9d5Sandi        }
6820cecf9d5Sandi
68307738714SAndreas Gohr        $address = $this->_xmlEntities($address);
68400a7b5adSEsther Brunner        $address = obfuscate($address);
68500a7b5adSEsther Brunner        $title   = $address;
68671352defSandi        if(empty($name)){
68700a7b5adSEsther Brunner            $name = $address;
68871352defSandi        }else{
689433bef32Sandi            $name = $this->_xmlEntities($name);
69071352defSandi        }
6910cecf9d5Sandi
692776b36ecSAndreas Gohr        if($conf['mailguard'] == 'visible') $address = rawurlencode($address);
693776b36ecSAndreas Gohr
694776b36ecSAndreas Gohr        $link['url']   = 'mailto:'.$address;
69571352defSandi        $link['name']  = $name;
69671352defSandi        $link['title'] = $title;
6970cecf9d5Sandi
69871352defSandi        //output formatted
699a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7000cecf9d5Sandi    }
7010cecf9d5Sandi
7024826ab45Sandi    function internalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
703dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
7043685f775Sandi        global $conf;
70537e34a5eSandi        global $ID;
70637e34a5eSandi        resolve_mediaid(getNS($ID),$src, $exists);
7070cecf9d5Sandi
7083685f775Sandi        $link = array();
7093685f775Sandi        $link['class']  = 'media';
7103685f775Sandi        $link['style']  = '';
7113685f775Sandi        $link['pre']    = '';
7123685f775Sandi        $link['suf']    = '';
7135e163278SAndreas Gohr        $link['more']   = '';
7143685f775Sandi        $link['target'] = $conf['target']['media'];
715d98d4540SBen Coburn        $noLink = false;
7163685f775Sandi
717433bef32Sandi        $link['title']  = $this->_xmlEntities($src);
71855efc227SAndreas Gohr        list($ext,$mime) = mimetype($src);
7195667eb65SAndreas Gohr        if(substr($mime,0,5) == 'image'){
720dc673a5bSjoe.lapp             $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),($linking=='direct'));
7212ca14335SEsther Brunner         }elseif($mime == 'application/x-shockwave-flash'){
7222ca14335SEsther Brunner             // don't link flash movies
7232ca14335SEsther Brunner             $noLink = TRUE;
72455efc227SAndreas Gohr         }else{
7252ca14335SEsther Brunner             // add file icons
7269d2ddea4SAndreas Gohr             $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
7279d2ddea4SAndreas Gohr             $link['class'] .= ' mediafile mf_'.$class;
7286de3759aSAndreas Gohr             $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),true);
72955efc227SAndreas Gohr         }
730433bef32Sandi         $link['name']   = $this->_media ($src, $title, $align, $width, $height, $cache);
7313685f775Sandi
7323685f775Sandi         //output formatted
733dc673a5bSjoe.lapp         if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
7342ca14335SEsther Brunner         else $this->doc .= $this->_formatLink($link);
7350cecf9d5Sandi    }
7360cecf9d5Sandi
7370cecf9d5Sandi    /**
7384826ab45Sandi     * @todo don't add link for flash
7390cecf9d5Sandi     */
7404826ab45Sandi    function externalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
741dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
7423685f775Sandi        global $conf;
7430cecf9d5Sandi
7443685f775Sandi        $link = array();
7453685f775Sandi        $link['class']  = 'media';
7463685f775Sandi        $link['style']  = '';
7473685f775Sandi        $link['pre']    = '';
7483685f775Sandi        $link['suf']    = '';
7495e163278SAndreas Gohr        $link['more']   = '';
7503685f775Sandi        $link['target'] = $conf['target']['media'];
7513685f775Sandi
752433bef32Sandi        $link['title']  = $this->_xmlEntities($src);
7536de3759aSAndreas Gohr        $link['url']    = ml($src,array('cache'=>$cache));
754433bef32Sandi        $link['name']   = $this->_media ($src, $title, $align, $width, $height, $cache);
755d98d4540SBen Coburn        $noLink = false;
7563685f775Sandi
7572ca14335SEsther Brunner        list($ext,$mime) = mimetype($src);
7582ca14335SEsther Brunner        if(substr($mime,0,5) == 'image'){
7592ca14335SEsther Brunner             // link only jpeg images
7602ca14335SEsther Brunner             // if ($ext != 'jpg' && $ext != 'jpeg') $noLink = TRUE;
7612ca14335SEsther Brunner        }elseif($mime == 'application/x-shockwave-flash'){
7622ca14335SEsther Brunner             // don't link flash movies
7632ca14335SEsther Brunner             $noLink = TRUE;
7642ca14335SEsther Brunner        }else{
7652ca14335SEsther Brunner             // add file icons
766d15166e5SAndreas Gohr             $link['class'] .= ' mediafile mf_'.$ext;
7672ca14335SEsther Brunner         }
7682ca14335SEsther Brunner
7693685f775Sandi        //output formatted
770dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
7712ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
7720cecf9d5Sandi    }
7730cecf9d5Sandi
7744826ab45Sandi    /**
7753db95becSAndreas Gohr     * Renders an RSS feed
776b625487dSandi     *
777b625487dSandi     * @author Andreas Gohr <andi@splitbrain.org>
778b625487dSandi     */
7793db95becSAndreas Gohr    function rss ($url,$params){
780b625487dSandi        global $lang;
7813db95becSAndreas Gohr        global $conf;
7823db95becSAndreas Gohr
7833db95becSAndreas Gohr        require_once(DOKU_INC.'inc/FeedParser.php');
7843db95becSAndreas Gohr        $feed = new FeedParser();
7853db95becSAndreas Gohr        $feed->feed_url($url);
786b625487dSandi
787b625487dSandi        //disable warning while fetching
788bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { $elvl = error_reporting(E_ERROR); }
7893db95becSAndreas Gohr        $rc = $feed->init();
790bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { error_reporting($elvl); }
791b625487dSandi
7923db95becSAndreas Gohr        //decide on start and end
7933db95becSAndreas Gohr        if($params['reverse']){
7943db95becSAndreas Gohr            $mod = -1;
7953db95becSAndreas Gohr            $start = $feed->get_item_quantity()-1;
7963db95becSAndreas Gohr            $end   = $start - ($params['max']);
797*b2a412b0SAndreas Gohr            $end   = ($end < -1) ? -1 : $end;
7983db95becSAndreas Gohr        }else{
7993db95becSAndreas Gohr            $mod   = 1;
8003db95becSAndreas Gohr            $start = 0;
8013db95becSAndreas Gohr            $end   = $feed->get_item_quantity();
8023db95becSAndreas Gohr            $end   = ($end > $params['max']) ? $params['max'] : $end;;
8033db95becSAndreas Gohr        }
8043db95becSAndreas Gohr
805a2d649c4Sandi        $this->doc .= '<ul class="rss">';
8063db95becSAndreas Gohr        if($rc){
8073db95becSAndreas Gohr            for ($x = $start; $x != $end; $x += $mod) {
8081bde1582SAndreas Gohr                $item = $feed->get_item($x);
8093db95becSAndreas Gohr                $this->doc .= '<li><div class="li">';
8101bde1582SAndreas Gohr                $this->externallink($item->get_permalink(),
8111bde1582SAndreas Gohr                                    $item->get_title());
8123db95becSAndreas Gohr                if($params['author']){
8131bde1582SAndreas Gohr                    $author = $item->get_author(0);
8141bde1582SAndreas Gohr                    if($author){
8151bde1582SAndreas Gohr                        $name = $author->get_name();
8161bde1582SAndreas Gohr                        if(!$name) $name = $author->get_email();
8171bde1582SAndreas Gohr                        if($name) $this->doc .= ' '.$lang['by'].' '.$name;
8181bde1582SAndreas Gohr                    }
8193db95becSAndreas Gohr                }
8203db95becSAndreas Gohr                if($params['date']){
8211bde1582SAndreas Gohr                    $this->doc .= ' ('.$item->get_date($conf['dformat']).')';
8223db95becSAndreas Gohr                }
8231bde1582SAndreas Gohr                if($params['details']){
8243db95becSAndreas Gohr                    $this->doc .= '<div class="detail">';
8253db95becSAndreas Gohr                    if($htmlok){
8261bde1582SAndreas Gohr                        $this->doc .= $item->get_description();
8273db95becSAndreas Gohr                    }else{
8281bde1582SAndreas Gohr                        $this->doc .= strip_tags($item->get_description());
8293db95becSAndreas Gohr                    }
8303db95becSAndreas Gohr                    $this->doc .= '</div>';
8313db95becSAndreas Gohr                }
8323db95becSAndreas Gohr
8333db95becSAndreas Gohr                $this->doc .= '</div></li>';
834b625487dSandi            }
835b625487dSandi        }else{
8363db95becSAndreas Gohr            $this->doc .= '<li><div class="li">';
837a2d649c4Sandi            $this->doc .= '<em>'.$lang['rssfailed'].'</em>';
838b625487dSandi            $this->externallink($url);
8393db95becSAndreas Gohr            $this->doc .= '</div></li>';
840b625487dSandi        }
841a2d649c4Sandi        $this->doc .= '</ul>';
842b625487dSandi    }
843b625487dSandi
8440cecf9d5Sandi    // $numrows not yet implemented
8450cecf9d5Sandi    function table_open($maxcols = NULL, $numrows = NULL){
846a2d649c4Sandi        $this->doc .= '<table class="inline">'.DOKU_LF;
8470cecf9d5Sandi    }
8480cecf9d5Sandi
8490cecf9d5Sandi    function table_close(){
850cb42b03dSAnika Henke        $this->doc .= '</table>'.DOKU_LF;
8510cecf9d5Sandi    }
8520cecf9d5Sandi
8530cecf9d5Sandi    function tablerow_open(){
854a2d649c4Sandi        $this->doc .= DOKU_TAB . '<tr>' . DOKU_LF . DOKU_TAB . DOKU_TAB;
8550cecf9d5Sandi    }
8560cecf9d5Sandi
8570cecf9d5Sandi    function tablerow_close(){
858a2d649c4Sandi        $this->doc .= DOKU_LF . DOKU_TAB . '</tr>' . DOKU_LF;
8590cecf9d5Sandi    }
8600cecf9d5Sandi
8610cecf9d5Sandi    function tableheader_open($colspan = 1, $align = NULL){
862a2d649c4Sandi        $this->doc .= '<th';
8630cecf9d5Sandi        if ( !is_null($align) ) {
864a2d649c4Sandi            $this->doc .= ' class="'.$align.'align"';
8650cecf9d5Sandi        }
8660cecf9d5Sandi        if ( $colspan > 1 ) {
867a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8680cecf9d5Sandi        }
869a2d649c4Sandi        $this->doc .= '>';
8700cecf9d5Sandi    }
8710cecf9d5Sandi
8720cecf9d5Sandi    function tableheader_close(){
873a2d649c4Sandi        $this->doc .= '</th>';
8740cecf9d5Sandi    }
8750cecf9d5Sandi
8760cecf9d5Sandi    function tablecell_open($colspan = 1, $align = NULL){
877a2d649c4Sandi        $this->doc .= '<td';
8780cecf9d5Sandi        if ( !is_null($align) ) {
879a2d649c4Sandi            $this->doc .= ' class="'.$align.'align"';
8800cecf9d5Sandi        }
8810cecf9d5Sandi        if ( $colspan > 1 ) {
882a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8830cecf9d5Sandi        }
884a2d649c4Sandi        $this->doc .= '>';
8850cecf9d5Sandi    }
8860cecf9d5Sandi
8870cecf9d5Sandi    function tablecell_close(){
888a2d649c4Sandi        $this->doc .= '</td>';
8890cecf9d5Sandi    }
8900cecf9d5Sandi
8910cecf9d5Sandi    //----------------------------------------------------------
8920cecf9d5Sandi    // Utils
8930cecf9d5Sandi
894ba11bd29Sandi    /**
8953fd0b676Sandi     * Build a link
8963fd0b676Sandi     *
8973fd0b676Sandi     * Assembles all parts defined in $link returns HTML for the link
898ba11bd29Sandi     *
899ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
900ba11bd29Sandi     */
901433bef32Sandi    function _formatLink($link){
902ba11bd29Sandi        //make sure the url is XHTML compliant (skip mailto)
903ba11bd29Sandi        if(substr($link['url'],0,7) != 'mailto:'){
904ba11bd29Sandi            $link['url'] = str_replace('&','&amp;',$link['url']);
905ba11bd29Sandi            $link['url'] = str_replace('&amp;amp;','&amp;',$link['url']);
906ba11bd29Sandi        }
907ba11bd29Sandi        //remove double encodings in titles
908ba11bd29Sandi        $link['title'] = str_replace('&amp;amp;','&amp;',$link['title']);
909ba11bd29Sandi
910453493f2SAndreas Gohr        // be sure there are no bad chars in url or title
911453493f2SAndreas Gohr        // (we can't do this for name because it can contain an img tag)
912453493f2SAndreas Gohr        $link['url']   = strtr($link['url'],array('>'=>'%3E','<'=>'%3C','"'=>'%22'));
913453493f2SAndreas Gohr        $link['title'] = strtr($link['title'],array('>'=>'&gt;','<'=>'&lt;','"'=>'&quot;'));
914453493f2SAndreas Gohr
915ba11bd29Sandi        $ret  = '';
916ba11bd29Sandi        $ret .= $link['pre'];
917ba11bd29Sandi        $ret .= '<a href="'.$link['url'].'"';
918bb4866bdSchris        if(!empty($link['class']))  $ret .= ' class="'.$link['class'].'"';
919bb4866bdSchris        if(!empty($link['target'])) $ret .= ' target="'.$link['target'].'"';
920bb4866bdSchris        if(!empty($link['title']))  $ret .= ' title="'.$link['title'].'"';
921bb4866bdSchris        if(!empty($link['style']))  $ret .= ' style="'.$link['style'].'"';
922bb4866bdSchris        if(!empty($link['more']))   $ret .= ' '.$link['more'];
923ba11bd29Sandi        $ret .= '>';
924ba11bd29Sandi        $ret .= $link['name'];
925ba11bd29Sandi        $ret .= '</a>';
926ba11bd29Sandi        $ret .= $link['suf'];
927ba11bd29Sandi        return $ret;
928ba11bd29Sandi    }
929ba11bd29Sandi
930ba11bd29Sandi    /**
931ba11bd29Sandi     * Removes any Namespace from the given name but keeps
932ba11bd29Sandi     * casing and special chars
933ba11bd29Sandi     *
934ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
935ba11bd29Sandi     */
936433bef32Sandi    function _simpleTitle($name){
937ba11bd29Sandi        global $conf;
938bd111079Sandi
939a966cbdeSAndreas Gohr        //if there is a hash we use the ancor name only
940a966cbdeSAndreas Gohr        list($name,$hash) = explode('#',$name,2);
941a966cbdeSAndreas Gohr        if($hash) return $hash;
942a966cbdeSAndreas Gohr
943a966cbdeSAndreas Gohr        //trim colons of a namespace link
944a966cbdeSAndreas Gohr        $name = rtrim($name,':');
945a966cbdeSAndreas Gohr
946ba11bd29Sandi        if($conf['useslash']){
947ba11bd29Sandi            $nssep = '[:;/]';
948ba11bd29Sandi        }else{
949ba11bd29Sandi            $nssep = '[:;]';
950ba11bd29Sandi        }
951bd111079Sandi        $name = preg_replace('!.*'.$nssep.'!','',$name);
952a966cbdeSAndreas Gohr
953a966cbdeSAndreas Gohr        if(!$name) return $this->_simpleTitle($conf['start']);
954bd111079Sandi        return $name;
955ba11bd29Sandi    }
956ba11bd29Sandi
9573fd0b676Sandi    /**
9583fd0b676Sandi     * Renders internal and external media
9593fd0b676Sandi     *
9603fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
9613fd0b676Sandi     */
9623fd0b676Sandi    function _media ($src, $title=NULL, $align=NULL, $width=NULL,
9633fd0b676Sandi                      $height=NULL, $cache=NULL) {
9643fd0b676Sandi
9653fd0b676Sandi        $ret = '';
9663fd0b676Sandi
9673fd0b676Sandi        list($ext,$mime) = mimetype($src);
9683fd0b676Sandi        if(substr($mime,0,5) == 'image'){
9693fd0b676Sandi            //add image tag
9706de3759aSAndreas Gohr            $ret .= '<img src="'.ml($src,array('w'=>$width,'h'=>$height,'cache'=>$cache)).'"';
9713fd0b676Sandi            $ret .= ' class="media'.$align.'"';
9723fd0b676Sandi
9733fd0b676Sandi            if (!is_null($title)) {
9743fd0b676Sandi                $ret .= ' title="'.$this->_xmlEntities($title).'"';
9753fd0b676Sandi                $ret .= ' alt="'.$this->_xmlEntities($title).'"';
97655efc227SAndreas Gohr            }elseif($ext == 'jpg' || $ext == 'jpeg'){
97755efc227SAndreas Gohr                //try to use the caption from IPTC/EXIF
97855efc227SAndreas Gohr                require_once(DOKU_INC.'inc/JpegMeta.php');
97955efc227SAndreas Gohr                $jpeg =& new JpegMeta(mediaFN($src));
98055efc227SAndreas Gohr                if($jpeg !== false) $cap = $jpeg->getTitle();
98155efc227SAndreas Gohr                if($cap){
98255efc227SAndreas Gohr                    $ret .= ' title="'.$this->_xmlEntities($cap).'"';
98355efc227SAndreas Gohr                    $ret .= ' alt="'.$this->_xmlEntities($cap).'"';
98455efc227SAndreas Gohr                }
9853fd0b676Sandi            }else{
9863fd0b676Sandi                $ret .= ' alt=""';
9873fd0b676Sandi            }
9883fd0b676Sandi
9893fd0b676Sandi            if ( !is_null($width) )
9903fd0b676Sandi                $ret .= ' width="'.$this->_xmlEntities($width).'"';
9913fd0b676Sandi
9923fd0b676Sandi            if ( !is_null($height) )
9933fd0b676Sandi                $ret .= ' height="'.$this->_xmlEntities($height).'"';
9943fd0b676Sandi
9953fd0b676Sandi            $ret .= ' />';
9963fd0b676Sandi
9973fd0b676Sandi        }elseif($mime == 'application/x-shockwave-flash'){
9983fd0b676Sandi            $ret .= '<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"'.
9993fd0b676Sandi                    ' codebase="http://download.macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version=6,0,40,0"';
10003fd0b676Sandi            if ( !is_null($width) ) $ret .= ' width="'.$this->_xmlEntities($width).'"';
10013fd0b676Sandi            if ( !is_null($height) ) $ret .= ' height="'.$this->_xmlEntities($height).'"';
10023fd0b676Sandi            $ret .= '>'.DOKU_LF;
10036de3759aSAndreas Gohr            $ret .= '<param name="movie" value="'.ml($src).'" />'.DOKU_LF;
10043fd0b676Sandi            $ret .= '<param name="quality" value="high" />'.DOKU_LF;
10056de3759aSAndreas Gohr            $ret .= '<embed src="'.ml($src).'"'.
10063fd0b676Sandi                    ' quality="high"';
10073fd0b676Sandi            if ( !is_null($width) ) $ret .= ' width="'.$this->_xmlEntities($width).'"';
10083fd0b676Sandi            if ( !is_null($height) ) $ret .= ' height="'.$this->_xmlEntities($height).'"';
10093fd0b676Sandi            $ret .= ' type="application/x-shockwave-flash"'.
10103fd0b676Sandi                    ' pluginspage="http://www.macromedia.com/go/getflashplayer"></embed>'.DOKU_LF;
10113fd0b676Sandi            $ret .= '</object>'.DOKU_LF;
10123fd0b676Sandi
10133fd0b676Sandi        }elseif(!is_null($title)){
10143fd0b676Sandi            // well at least we have a title to display
10153fd0b676Sandi            $ret .= $this->_xmlEntities($title);
10163fd0b676Sandi        }else{
10175291ca3aSAndreas Gohr            // just show the sourcename
10185291ca3aSAndreas Gohr            $ret .= $this->_xmlEntities(noNS($src));
10193fd0b676Sandi        }
10203fd0b676Sandi
10213fd0b676Sandi        return $ret;
10223fd0b676Sandi    }
10233fd0b676Sandi
1024433bef32Sandi    function _xmlEntities($string) {
10250cecf9d5Sandi        return htmlspecialchars($string);
10260cecf9d5Sandi    }
10270cecf9d5Sandi
10288a831f2bSAndreas Gohr    /**
10298a831f2bSAndreas Gohr     * Creates a linkid from a headline
1030c5a8fd96SAndreas Gohr     *
1031c5a8fd96SAndreas Gohr     * @param string  $title   The headline title
1032c5a8fd96SAndreas Gohr     * @param boolean $create  Create a new unique ID?
1033c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
10348a831f2bSAndreas Gohr     */
1035c5a8fd96SAndreas Gohr    function _headerToLink($title,$create=false) {
1036501af51eSAndreas Gohr        $title = str_replace(':','',cleanID($title));
1037a9e0fd1bSAndreas Gohr        $title = ltrim($title,'0123456789._-');
10388a831f2bSAndreas Gohr        if(empty($title)) $title='section';
1039c5a8fd96SAndreas Gohr
1040c5a8fd96SAndreas Gohr        if($create){
1041c5a8fd96SAndreas Gohr            // make sure tiles are unique
1042c5a8fd96SAndreas Gohr            $num = '';
1043c5a8fd96SAndreas Gohr            while(in_array($title.$num,$this->headers)){
10440affd488SAndreas Gohr                ($num) ? $num++ : $num = 1;
1045c5a8fd96SAndreas Gohr            }
1046c5a8fd96SAndreas Gohr            $title = $title.$num;
1047c5a8fd96SAndreas Gohr            $this->headers[] = $title;
1048c5a8fd96SAndreas Gohr        }
1049c5a8fd96SAndreas Gohr
10508a831f2bSAndreas Gohr        return $title;
10510cecf9d5Sandi    }
10520cecf9d5Sandi
1053af587fa8Sandi    /**
10543fd0b676Sandi     * Construct a title and handle images in titles
10553fd0b676Sandi     *
10560b7c14c2Sandi     * @author Harry Fuecks <hfuecks@gmail.com>
10573fd0b676Sandi     */
1058433bef32Sandi    function _getLinkTitle($title, $default, & $isImage, $id=NULL) {
1059bb0a59d4Sjan        global $conf;
1060bb0a59d4Sjan
10610cecf9d5Sandi        $isImage = FALSE;
10620cecf9d5Sandi        if ( is_null($title) ) {
1063bb0a59d4Sjan            if ($conf['useheading'] && $id) {
1064bb0a59d4Sjan                $heading = p_get_first_heading($id);
1065bb0a59d4Sjan                if ($heading) {
1066433bef32Sandi                    return $this->_xmlEntities($heading);
1067bb0a59d4Sjan                }
1068bb0a59d4Sjan            }
1069433bef32Sandi            return $this->_xmlEntities($default);
10700cecf9d5Sandi        } else if ( is_string($title) ) {
1071433bef32Sandi            return $this->_xmlEntities($title);
10720cecf9d5Sandi        } else if ( is_array($title) ) {
10730cecf9d5Sandi            $isImage = TRUE;
1074433bef32Sandi            return $this->_imageTitle($title);
10750cecf9d5Sandi        }
10760cecf9d5Sandi    }
10770cecf9d5Sandi
10780cecf9d5Sandi    /**
10793fd0b676Sandi     * Returns an HTML code for images used in link titles
10803fd0b676Sandi     *
10813fd0b676Sandi     * @todo Resolve namespace on internal images
10823fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
10830cecf9d5Sandi     */
1084433bef32Sandi    function _imageTitle($img) {
1085433bef32Sandi        return $this->_media($img['src'],
10864826ab45Sandi                              $img['title'],
10874826ab45Sandi                              $img['align'],
10884826ab45Sandi                              $img['width'],
10894826ab45Sandi                              $img['height'],
10904826ab45Sandi                              $img['cache']);
10910cecf9d5Sandi    }
10920cecf9d5Sandi}
10930cecf9d5Sandi
10944826ab45Sandi//Setup VIM: ex: et ts=4 enc=utf-8 :
1095