xref: /dokuwiki/inc/parser/xhtml.php (revision e41c4da9b8ec98a022a91b04f7c5d7bbee54b6c9)
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';
220e1c636eSandi
230cecf9d5Sandi/**
24b625487dSandi * The Renderer
250cecf9d5Sandi */
26ac83b9d8Sandiclass Doku_Renderer_xhtml extends Doku_Renderer {
270cecf9d5Sandi
28c5a8fd96SAndreas Gohr    // @access public
29c5a8fd96SAndreas Gohr    var $doc = '';        // will contain the whole document
30c5a8fd96SAndreas Gohr    var $toc = array();   // will contain the Table of Contents
31c5a8fd96SAndreas Gohr
320cecf9d5Sandi
330cecf9d5Sandi    var $headers = array();
340cecf9d5Sandi
350cecf9d5Sandi    var $footnotes = array();
360cecf9d5Sandi
370cecf9d5Sandi    var $acronyms = array();
380cecf9d5Sandi    var $smileys = array();
390cecf9d5Sandi    var $badwords = array();
400cecf9d5Sandi    var $entities = array();
410cecf9d5Sandi    var $interwiki = array();
420cecf9d5Sandi
43af587fa8Sandi    var $lastsec = 0;
44af587fa8Sandi
457764a90aSandi    var $store = '';
467764a90aSandi
470cecf9d5Sandi    function document_start() {
48c5a8fd96SAndreas Gohr        //reset some internals
49c5a8fd96SAndreas Gohr        $this->toc     = array();
50c5a8fd96SAndreas Gohr        $this->headers = array();
510cecf9d5Sandi    }
520cecf9d5Sandi
530cecf9d5Sandi    function document_end() {
541d47afe1Sandi        // add button for last section if any and more than one
55433bef32Sandi        if($this->lastsec > 1) $this->_secedit($this->lastsec,'');
560cecf9d5Sandi
570cecf9d5Sandi        if ( count ($this->footnotes) > 0 ) {
58a2d649c4Sandi            $this->doc .= '<div class="footnotes">'.DOKU_LF;
59d74aace9Schris
60d74aace9Schris            $id = 0;
610cecf9d5Sandi            foreach ( $this->footnotes as $footnote ) {
62d74aace9Schris                $id++;   // the number of the current footnote
63d74aace9Schris
64d74aace9Schris                // check its not a placeholder that indicates actual footnote text is elsewhere
65d74aace9Schris                if (substr($footnote, 0, 5) != "@@FNT") {
66d74aace9Schris
67d74aace9Schris                    // open the footnote and set the anchor and backlink
68d74aace9Schris                    $this->doc .= '<div class="fn">';
69d74aace9Schris                    $this->doc .= '<a href="#fnt'.$id.'" id="fn'.$id.'" name="fn'.$id.'" class="fn_bot">';
70d74aace9Schris                    $this->doc .= $id.')</a> '.DOKU_LF;
71d74aace9Schris
72d74aace9Schris                    // get any other footnotes that use the same markup
73d74aace9Schris                    $alt = array_keys($this->footnotes, "@@FNT$id");
74d74aace9Schris
75d74aace9Schris                    if (count($alt)) {
76d74aace9Schris                      foreach ($alt as $ref) {
77d74aace9Schris                        // set anchor and backlink for the other footnotes
78d74aace9Schris                        $this->doc .= ', <a href="#fnt'.($ref+1).'" id="fn'.($ref+1).'" name="fn'.($ref+1).'" class="fn_bot">';
79d74aace9Schris                        $this->doc .= ($ref+1).')</a> '.DOKU_LF;
80d74aace9Schris                      }
81d74aace9Schris                    }
82d74aace9Schris
83d74aace9Schris                    // add footnote markup and close this footnote
84a2d649c4Sandi                    $this->doc .= $footnote;
85d74aace9Schris                    $this->doc .= '</div>' . DOKU_LF;
86d74aace9Schris                }
870cecf9d5Sandi            }
88a2d649c4Sandi            $this->doc .= '</div>'.DOKU_LF;
890cecf9d5Sandi        }
90c5a8fd96SAndreas Gohr
91c5a8fd96SAndreas Gohr        // prepend the TOC
92*e41c4da9SAndreas Gohr        if($this->info['toc']){
93c5a8fd96SAndreas Gohr            $this->doc = $this->render_TOC().$this->doc;
940cecf9d5Sandi        }
95*e41c4da9SAndreas Gohr    }
960cecf9d5Sandi
97c5a8fd96SAndreas Gohr    /**
98c5a8fd96SAndreas Gohr     * Return the TOC rendered to XHTML
99c5a8fd96SAndreas Gohr     *
100c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
101c5a8fd96SAndreas Gohr     */
102c5a8fd96SAndreas Gohr    function render_TOC(){
103*e41c4da9SAndreas Gohr        if(count($this->toc) < 3) return '';
104c5a8fd96SAndreas Gohr        global $lang;
105c5a8fd96SAndreas Gohr        $out  = '<div class="toc">'.DOKU_LF;
106c5a8fd96SAndreas Gohr        $out .= '<div class="tocheader toctoggle" id="toc__header">';
107c5a8fd96SAndreas Gohr        $out .= $lang['toc'];
108c5a8fd96SAndreas Gohr        $out .= '</div>'.DOKU_LF;
109c5a8fd96SAndreas Gohr        $out .= '<div id="toc__inside">'.DOKU_LF;
110c5a8fd96SAndreas Gohr        $out .= html_buildlist($this->toc,'toc',array($this,'_tocitem'));
111c5a8fd96SAndreas Gohr        $out .= '</div>'.DOKU_LF.'</div>'.DOKU_LF;
112c5a8fd96SAndreas Gohr        return $out;
113c5a8fd96SAndreas Gohr    }
114c5a8fd96SAndreas Gohr
115c5a8fd96SAndreas Gohr    /**
116c5a8fd96SAndreas Gohr     * Callback for html_buildlist
117c5a8fd96SAndreas Gohr     */
118c5a8fd96SAndreas Gohr    function _tocitem($item){
119c5a8fd96SAndreas Gohr        return '<span class="li"><a href="#'.$item['hid'].'" class="toc">'.
120c5a8fd96SAndreas Gohr               $this->_xmlEntities($item['title']).'</a></span>';
121c5a8fd96SAndreas Gohr    }
122c5a8fd96SAndreas Gohr
123c5a8fd96SAndreas Gohr/** FIXME deprecated
124c5a8fd96SAndreas Gohr
1250cecf9d5Sandi    function toc_open() {
126d35041baSandi        global $lang;
127a2d649c4Sandi        $this->doc .= '<div class="toc">'.DOKU_LF;
1284ce6ec08SAndreas Gohr        $this->doc .= '<div class="tocheader toctoggle" id="toc__header">';
129d35041baSandi        $this->doc .= $lang['toc'];
130e4a9930bSmatthiasgrimm        $this->doc .= '</div>'.DOKU_LF;
13100540a38SAndreas Gohr        $this->doc .= '<div id="toc__inside">'.DOKU_LF;
1320cecf9d5Sandi    }
1330cecf9d5Sandi
1340cecf9d5Sandi    function tocbranch_open($level) {
135a2d649c4Sandi        $this->doc .= '<ul class="toc">'.DOKU_LF;
1360cecf9d5Sandi    }
1370cecf9d5Sandi
1380cecf9d5Sandi    function tocitem_open($level, $empty = FALSE) {
1390cecf9d5Sandi        if ( !$empty ) {
140a2d649c4Sandi            $this->doc .= '<li class="level'.$level.'">';
1410cecf9d5Sandi        } else {
142a2d649c4Sandi            $this->doc .= '<li class="clear">';
1430cecf9d5Sandi        }
1440cecf9d5Sandi    }
1450cecf9d5Sandi
1460cecf9d5Sandi    function tocelement($level, $title) {
147a2d649c4Sandi        $this->doc .= '<span class="li"><a href="#'.$this->_headerToLink($title).'" class="toc">';
148a2d649c4Sandi        $this->doc .= $this->_xmlEntities($title);
149a2d649c4Sandi        $this->doc .= '</a></span>';
1500cecf9d5Sandi    }
1510cecf9d5Sandi
1520cecf9d5Sandi    function tocitem_close($level) {
153a2d649c4Sandi        $this->doc .= '</li>'.DOKU_LF;
1540cecf9d5Sandi    }
1550cecf9d5Sandi
1560cecf9d5Sandi    function tocbranch_close($level) {
157a2d649c4Sandi        $this->doc .= '</ul>'.DOKU_LF;
1580cecf9d5Sandi    }
1590cecf9d5Sandi
1600cecf9d5Sandi    function toc_close() {
161a2d649c4Sandi        $this->doc .= '</div>'.DOKU_LF.'</div>'.DOKU_LF;
1620cecf9d5Sandi    }
1630cecf9d5Sandi
164c5a8fd96SAndreas Gohr*/
165c5a8fd96SAndreas Gohr
166af587fa8Sandi    function header($text, $level, $pos) {
167af587fa8Sandi        global $conf;
168af587fa8Sandi        //handle section editing
169af587fa8Sandi        if($level <= $conf['maxseclevel']){
170af587fa8Sandi            // add button for last section if any
171433bef32Sandi            if($this->lastsec) $this->_secedit($this->lastsec,$pos-1);
172af587fa8Sandi            // remember current position
173af587fa8Sandi            $this->lastsec = $pos;
174af587fa8Sandi        }
175af587fa8Sandi
176c5a8fd96SAndreas Gohr        // create a unique header id
177c5a8fd96SAndreas Gohr        $hid = $this->_headerToLink($text,'true');
178c5a8fd96SAndreas Gohr
179c5a8fd96SAndreas Gohr        //handle TOC
180c5a8fd96SAndreas Gohr        if($level >= $conf['toptoclevel'] && $level <= $conf['maxtoclevel']){
181c5a8fd96SAndreas Gohr            // the TOC is one of our standard ul list arrays ;-)
182c5a8fd96SAndreas Gohr            $this->toc[] = array( 'hid'   => $hid,
183c5a8fd96SAndreas Gohr                                  'title' => $text,
184c5a8fd96SAndreas Gohr                                  'type'  => 'ul',
185c5a8fd96SAndreas Gohr                                  'level' => $level);
186c5a8fd96SAndreas Gohr        }
187c5a8fd96SAndreas Gohr
188c5a8fd96SAndreas Gohr        // write the header
189c5a8fd96SAndreas Gohr        $this->doc .= DOKU_LF.'<h'.$level.'><a name="'.$hid.'" id="'.$hid.'">';
190a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
191c6a1c1bfSAnika Henke        $this->doc .= "</a></h$level>".DOKU_LF;
1920cecf9d5Sandi    }
1930cecf9d5Sandi
1940cecf9d5Sandi    function section_open($level) {
195a2d649c4Sandi        $this->doc .= "<div class=\"level$level\">".DOKU_LF;
1960cecf9d5Sandi    }
1970cecf9d5Sandi
1980cecf9d5Sandi    function section_close() {
199a2d649c4Sandi        $this->doc .= DOKU_LF.'</div>'.DOKU_LF;
2000cecf9d5Sandi    }
2010cecf9d5Sandi
2020cecf9d5Sandi    function cdata($text) {
203a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
2040cecf9d5Sandi    }
2050cecf9d5Sandi
2060cecf9d5Sandi    function p_open() {
207a2d649c4Sandi        $this->doc .= DOKU_LF.'<p>'.DOKU_LF;
2080cecf9d5Sandi    }
2090cecf9d5Sandi
2100cecf9d5Sandi    function p_close() {
211a2d649c4Sandi        $this->doc .= DOKU_LF.'</p>'.DOKU_LF;
2120cecf9d5Sandi    }
2130cecf9d5Sandi
2140cecf9d5Sandi    function linebreak() {
215a2d649c4Sandi        $this->doc .= '<br/>'.DOKU_LF;
2160cecf9d5Sandi    }
2170cecf9d5Sandi
2180cecf9d5Sandi    function hr() {
2194beabca9SAnika Henke        $this->doc .= '<hr />'.DOKU_LF;
2200cecf9d5Sandi    }
2210cecf9d5Sandi
2220cecf9d5Sandi    function strong_open() {
223a2d649c4Sandi        $this->doc .= '<strong>';
2240cecf9d5Sandi    }
2250cecf9d5Sandi
2260cecf9d5Sandi    function strong_close() {
227a2d649c4Sandi        $this->doc .= '</strong>';
2280cecf9d5Sandi    }
2290cecf9d5Sandi
2300cecf9d5Sandi    function emphasis_open() {
231a2d649c4Sandi        $this->doc .= '<em>';
2320cecf9d5Sandi    }
2330cecf9d5Sandi
2340cecf9d5Sandi    function emphasis_close() {
235a2d649c4Sandi        $this->doc .= '</em>';
2360cecf9d5Sandi    }
2370cecf9d5Sandi
2380cecf9d5Sandi    function underline_open() {
23902e51121SAnika Henke        $this->doc .= '<em class="u">';
2400cecf9d5Sandi    }
2410cecf9d5Sandi
2420cecf9d5Sandi    function underline_close() {
24302e51121SAnika Henke        $this->doc .= '</em>';
2440cecf9d5Sandi    }
2450cecf9d5Sandi
2460cecf9d5Sandi    function monospace_open() {
247a2d649c4Sandi        $this->doc .= '<code>';
2480cecf9d5Sandi    }
2490cecf9d5Sandi
2500cecf9d5Sandi    function monospace_close() {
251a2d649c4Sandi        $this->doc .= '</code>';
2520cecf9d5Sandi    }
2530cecf9d5Sandi
2540cecf9d5Sandi    function subscript_open() {
255a2d649c4Sandi        $this->doc .= '<sub>';
2560cecf9d5Sandi    }
2570cecf9d5Sandi
2580cecf9d5Sandi    function subscript_close() {
259a2d649c4Sandi        $this->doc .= '</sub>';
2600cecf9d5Sandi    }
2610cecf9d5Sandi
2620cecf9d5Sandi    function superscript_open() {
263a2d649c4Sandi        $this->doc .= '<sup>';
2640cecf9d5Sandi    }
2650cecf9d5Sandi
2660cecf9d5Sandi    function superscript_close() {
267a2d649c4Sandi        $this->doc .= '</sup>';
2680cecf9d5Sandi    }
2690cecf9d5Sandi
2700cecf9d5Sandi    function deleted_open() {
271a2d649c4Sandi        $this->doc .= '<del>';
2720cecf9d5Sandi    }
2730cecf9d5Sandi
2740cecf9d5Sandi    function deleted_close() {
275a2d649c4Sandi        $this->doc .= '</del>';
2760cecf9d5Sandi    }
2770cecf9d5Sandi
2783fd0b676Sandi    /**
2793fd0b676Sandi     * Callback for footnote start syntax
2803fd0b676Sandi     *
2813fd0b676Sandi     * All following content will go to the footnote instead of
282d74aace9Schris     * the document. To achieve this the previous rendered content
2833fd0b676Sandi     * is moved to $store and $doc is cleared
2843fd0b676Sandi     *
2853fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
2863fd0b676Sandi     */
2870cecf9d5Sandi    function footnote_open() {
2887764a90aSandi
2897764a90aSandi        // move current content to store and record footnote
2907764a90aSandi        $this->store = $this->doc;
2917764a90aSandi        $this->doc   = '';
2920cecf9d5Sandi    }
2930cecf9d5Sandi
2943fd0b676Sandi    /**
2953fd0b676Sandi     * Callback for footnote end syntax
2963fd0b676Sandi     *
2973fd0b676Sandi     * All rendered content is moved to the $footnotes array and the old
2983fd0b676Sandi     * content is restored from $store again
2993fd0b676Sandi     *
3003fd0b676Sandi     * @author Andreas Gohr
3013fd0b676Sandi     */
3020cecf9d5Sandi    function footnote_close() {
3037764a90aSandi
304d74aace9Schris        // recover footnote into the stack and restore old content
305d74aace9Schris        $footnote = $this->doc;
3067764a90aSandi        $this->doc = $this->store;
3077764a90aSandi        $this->store = '';
308d74aace9Schris
309d74aace9Schris        // check to see if this footnote has been seen before
310d74aace9Schris        $i = array_search($footnote, $this->footnotes);
311d74aace9Schris
312d74aace9Schris        if ($i === false) {
313d74aace9Schris            // its a new footnote, add it to the $footnotes array
314d74aace9Schris            $id = count($this->footnotes)+1;
315d74aace9Schris            $this->footnotes[count($this->footnotes)] = $footnote;
316d74aace9Schris        } else {
317d74aace9Schris            // seen this one before, translate the index to an id and save a placeholder
318d74aace9Schris            $i++;
319d74aace9Schris            $id = count($this->footnotes)+1;
320d74aace9Schris            $this->footnotes[count($this->footnotes)] = "@@FNT".($i);
321d74aace9Schris        }
322d74aace9Schris
323d74aace9Schris        // output the footnote reference and link, incl. onmouseover for insitu footnote popup
32402e51121SAnika Henke        $this->doc .= '<a href="#fn'.$id.'" name="fnt'.$id.'" id="fnt'.$id.'" class="fn_top" onmouseover="fnt(\''.$id.'\', this, event);">'.$id.')</a>';
3250cecf9d5Sandi    }
3260cecf9d5Sandi
3270cecf9d5Sandi    function listu_open() {
328a2d649c4Sandi        $this->doc .= '<ul>'.DOKU_LF;
3290cecf9d5Sandi    }
3300cecf9d5Sandi
3310cecf9d5Sandi    function listu_close() {
332a2d649c4Sandi        $this->doc .= '</ul>'.DOKU_LF;
3330cecf9d5Sandi    }
3340cecf9d5Sandi
3350cecf9d5Sandi    function listo_open() {
336a2d649c4Sandi        $this->doc .= '<ol>'.DOKU_LF;
3370cecf9d5Sandi    }
3380cecf9d5Sandi
3390cecf9d5Sandi    function listo_close() {
340a2d649c4Sandi        $this->doc .= '</ol>'.DOKU_LF;
3410cecf9d5Sandi    }
3420cecf9d5Sandi
3430cecf9d5Sandi    function listitem_open($level) {
344a2d649c4Sandi        $this->doc .= '<li class="level'.$level.'">';
3450cecf9d5Sandi    }
3460cecf9d5Sandi
3470cecf9d5Sandi    function listitem_close() {
348a2d649c4Sandi        $this->doc .= '</li>'.DOKU_LF;
3490cecf9d5Sandi    }
3500cecf9d5Sandi
3510cecf9d5Sandi    function listcontent_open() {
35290db23d7Schris        $this->doc .= '<div class="li">';
3530cecf9d5Sandi    }
3540cecf9d5Sandi
3550cecf9d5Sandi    function listcontent_close() {
35690db23d7Schris        $this->doc .= '</div>'.DOKU_LF;
3570cecf9d5Sandi    }
3580cecf9d5Sandi
3590cecf9d5Sandi    function unformatted($text) {
360a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
3610cecf9d5Sandi    }
3620cecf9d5Sandi
3630cecf9d5Sandi    /**
3643fd0b676Sandi     * Execute PHP code if allowed
3653fd0b676Sandi     *
3663fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3670cecf9d5Sandi     */
3680cecf9d5Sandi    function php($text) {
3694de671bcSandi        global $conf;
3704de671bcSandi        if($conf['phpok']){
371bad0b545Sandi            ob_start();
3724de671bcSandi            eval($text);
3733fd0b676Sandi            $this->doc .= ob_get_contents();
374bad0b545Sandi            ob_end_clean();
3754de671bcSandi        }else{
3764de671bcSandi            $this->file($text);
3774de671bcSandi        }
3780cecf9d5Sandi    }
3790cecf9d5Sandi
3800cecf9d5Sandi    /**
3813fd0b676Sandi     * Insert HTML if allowed
3823fd0b676Sandi     *
3833fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3840cecf9d5Sandi     */
3850cecf9d5Sandi    function html($text) {
3864de671bcSandi        global $conf;
3874de671bcSandi        if($conf['htmlok']){
388a2d649c4Sandi          $this->doc .= $text;
3894de671bcSandi        }else{
3900cecf9d5Sandi          $this->file($text);
3910cecf9d5Sandi        }
3924de671bcSandi    }
3930cecf9d5Sandi
3940cecf9d5Sandi    function preformatted($text) {
395a2d649c4Sandi        $this->doc .= '<pre class="code">' . $this->_xmlEntities($text) . '</pre>'. DOKU_LF;
3960cecf9d5Sandi    }
3970cecf9d5Sandi
3980cecf9d5Sandi    function file($text) {
399a2d649c4Sandi        $this->doc .= '<pre class="file">' . $this->_xmlEntities($text). '</pre>'. DOKU_LF;
4000cecf9d5Sandi    }
4010cecf9d5Sandi
4020cecf9d5Sandi    function quote_open() {
40396331712SAnika Henke        $this->doc .= '<blockquote><div class="no">'.DOKU_LF;
4040cecf9d5Sandi    }
4050cecf9d5Sandi
4060cecf9d5Sandi    function quote_close() {
40796331712SAnika Henke        $this->doc .= '</div></blockquote>'.DOKU_LF;
4080cecf9d5Sandi    }
4090cecf9d5Sandi
4100cecf9d5Sandi    /**
4113fd0b676Sandi     * Callback for code text
4123fd0b676Sandi     *
4133fd0b676Sandi     * Uses GeSHi to highlight language syntax
4143fd0b676Sandi     *
4153fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
4160cecf9d5Sandi     */
4170cecf9d5Sandi    function code($text, $language = NULL) {
4184de671bcSandi        global $conf;
4190cecf9d5Sandi
4200cecf9d5Sandi        if ( is_null($language) ) {
4210cecf9d5Sandi            $this->preformatted($text);
4220cecf9d5Sandi        } else {
423313da78aSandi            //strip leading blank line
424313da78aSandi            $text = preg_replace('/^\s*?\n/','',$text);
425313da78aSandi            // Handle with Geshi here
4267196b541Sandi            require_once(DOKU_INC . 'inc/geshi.php');
4277196b541Sandi            $geshi = new GeSHi($text, strtolower($language), DOKU_INC . 'inc/geshi');
428313da78aSandi            $geshi->set_encoding('utf-8');
4290cecf9d5Sandi            $geshi->enable_classes();
4300cecf9d5Sandi            $geshi->set_header_type(GESHI_HEADER_PRE);
431750fd6a8SChristopher Smith            $geshi->set_overall_class("code $language");
4324de671bcSandi            $geshi->set_link_target($conf['target']['extern']);
4330cecf9d5Sandi
4340cecf9d5Sandi            $text = $geshi->parse_code();
435a2d649c4Sandi            $this->doc .= $text;
4360cecf9d5Sandi        }
4370cecf9d5Sandi    }
4380cecf9d5Sandi
4390cecf9d5Sandi    function acronym($acronym) {
4400cecf9d5Sandi
4410cecf9d5Sandi        if ( array_key_exists($acronym, $this->acronyms) ) {
4420cecf9d5Sandi
443433bef32Sandi            $title = $this->_xmlEntities($this->acronyms[$acronym]);
4440cecf9d5Sandi
445a2d649c4Sandi            $this->doc .= '<acronym title="'.$title
446433bef32Sandi                .'">'.$this->_xmlEntities($acronym).'</acronym>';
4470cecf9d5Sandi
4480cecf9d5Sandi        } else {
449a2d649c4Sandi            $this->doc .= $this->_xmlEntities($acronym);
4500cecf9d5Sandi        }
4510cecf9d5Sandi    }
4520cecf9d5Sandi
4530cecf9d5Sandi    function smiley($smiley) {
4540cecf9d5Sandi        if ( array_key_exists($smiley, $this->smileys) ) {
455433bef32Sandi            $title = $this->_xmlEntities($this->smileys[$smiley]);
456f62ea8a1Sandi            $this->doc .= '<img src="'.DOKU_BASE.'lib/images/smileys/'.$this->smileys[$smiley].
4574beabca9SAnika Henke                '" class="middle" alt="'.
458433bef32Sandi                    $this->_xmlEntities($smiley).'" />';
4590cecf9d5Sandi        } else {
460a2d649c4Sandi            $this->doc .= $this->_xmlEntities($smiley);
4610cecf9d5Sandi        }
4620cecf9d5Sandi    }
4630cecf9d5Sandi
464f62ea8a1Sandi    /*
4654de671bcSandi    * not used
4660cecf9d5Sandi    function wordblock($word) {
4670cecf9d5Sandi        if ( array_key_exists($word, $this->badwords) ) {
468a2d649c4Sandi            $this->doc .= '** BLEEP **';
4690cecf9d5Sandi        } else {
470a2d649c4Sandi            $this->doc .= $this->_xmlEntities($word);
4710cecf9d5Sandi        }
4720cecf9d5Sandi    }
4734de671bcSandi    */
4740cecf9d5Sandi
4750cecf9d5Sandi    function entity($entity) {
4760cecf9d5Sandi        if ( array_key_exists($entity, $this->entities) ) {
477a2d649c4Sandi            $this->doc .= $this->entities[$entity];
4780cecf9d5Sandi        } else {
479a2d649c4Sandi            $this->doc .= $this->_xmlEntities($entity);
4800cecf9d5Sandi        }
4810cecf9d5Sandi    }
4820cecf9d5Sandi
4830cecf9d5Sandi    function multiplyentity($x, $y) {
484a2d649c4Sandi        $this->doc .= "$x&times;$y";
4850cecf9d5Sandi    }
4860cecf9d5Sandi
4870cecf9d5Sandi    function singlequoteopening() {
488a2d649c4Sandi        $this->doc .= "&lsquo;";
4890cecf9d5Sandi    }
4900cecf9d5Sandi
4910cecf9d5Sandi    function singlequoteclosing() {
492a2d649c4Sandi        $this->doc .= "&rsquo;";
4930cecf9d5Sandi    }
4940cecf9d5Sandi
4950cecf9d5Sandi    function doublequoteopening() {
496a2d649c4Sandi        $this->doc .= "&ldquo;";
4970cecf9d5Sandi    }
4980cecf9d5Sandi
4990cecf9d5Sandi    function doublequoteclosing() {
500a2d649c4Sandi        $this->doc .= "&rdquo;";
5010cecf9d5Sandi    }
5020cecf9d5Sandi
5030cecf9d5Sandi    /**
5040cecf9d5Sandi    */
5050cecf9d5Sandi    function camelcaselink($link) {
50611d0aa47Sandi      $this->internallink($link,$link);
5070cecf9d5Sandi    }
5080cecf9d5Sandi
5090b7c14c2Sandi
5100b7c14c2Sandi    function locallink($hash, $name = NULL){
5110b7c14c2Sandi        global $ID;
5120b7c14c2Sandi        $name  = $this->_getLinkTitle($name, $hash, $isImage);
5130b7c14c2Sandi        $hash  = $this->_headerToLink($hash);
5140b7c14c2Sandi        $title = $ID.' &crarr;';
5150b7c14c2Sandi        $this->doc .= '<a href="#'.$hash.'" title="'.$title.'" class="wikilink1">';
5160b7c14c2Sandi        $this->doc .= $name;
5170b7c14c2Sandi        $this->doc .= '</a>';
5180b7c14c2Sandi    }
5190b7c14c2Sandi
520cffcc403Sandi    /**
5213fd0b676Sandi     * Render an internal Wiki Link
5223fd0b676Sandi     *
523cffcc403Sandi     * $search and $returnonly are not for the renderer but are used
524cffcc403Sandi     * elsewhere - no need to implement them in other renderers
5253fd0b676Sandi     *
5263fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
527cffcc403Sandi     */
528cffcc403Sandi    function internallink($id, $name = NULL, $search=NULL,$returnonly=false) {
529ba11bd29Sandi        global $conf;
53037e34a5eSandi        global $ID;
5310339c872Sjan        // default name is based on $id as given
5320339c872Sjan        $default = $this->_simpleTitle($id);
5330339c872Sjan        // now first resolve and clean up the $id
53437e34a5eSandi        resolve_pageid(getNS($ID),$id,$exists);
5350339c872Sjan        $name = $this->_getLinkTitle($name, $default, $isImage, $id);
5360e1c636eSandi        if ( !$isImage ) {
5370e1c636eSandi            if ( $exists ) {
538ba11bd29Sandi                $class='wikilink1';
5390cecf9d5Sandi            } else {
540ba11bd29Sandi                $class='wikilink2';
5410cecf9d5Sandi            }
5420cecf9d5Sandi        } else {
543ba11bd29Sandi            $class='media';
5440cecf9d5Sandi        }
5450cecf9d5Sandi
546a1685bedSandi        //keep hash anchor
547a1685bedSandi        list($id,$hash) = split('#',$id,2);
548a1685bedSandi
549ba11bd29Sandi        //prepare for formating
550ba11bd29Sandi        $link['target'] = $conf['target']['wiki'];
551ba11bd29Sandi        $link['style']  = '';
552ba11bd29Sandi        $link['pre']    = '';
553ba11bd29Sandi        $link['suf']    = '';
55440eb54bbSjan        // highlight link to current page
55540eb54bbSjan        if ($id == $ID) {
55692795d04Sandi            $link['pre']    = '<span class="curid">';
55792795d04Sandi            $link['suf']    = '</span>';
55840eb54bbSjan        }
5595e163278SAndreas Gohr        $link['more']   = '';
560ba11bd29Sandi        $link['class']  = $class;
561ba11bd29Sandi        $link['url']    = wl($id);
562ba11bd29Sandi        $link['name']   = $name;
563ba11bd29Sandi        $link['title']  = $id;
564723d78dbSandi        //add search string
565723d78dbSandi        if($search){
566723d78dbSandi            ($conf['userewrite']) ? $link['url'].='?s=' : $link['url'].='&amp;s=';
567b6c6979fSAndreas Gohr            $link['url'] .= rawurlencode($search);
568723d78dbSandi        }
569723d78dbSandi
570a1685bedSandi        //keep hash
571a1685bedSandi        if($hash) $link['url'].='#'.$hash;
572a1685bedSandi
573ba11bd29Sandi        //output formatted
574cffcc403Sandi        if($returnonly){
575cffcc403Sandi            return $this->_formatLink($link);
576cffcc403Sandi        }else{
577a2d649c4Sandi            $this->doc .= $this->_formatLink($link);
5780cecf9d5Sandi        }
579cffcc403Sandi    }
5800cecf9d5Sandi
581b625487dSandi    function externallink($url, $name = NULL) {
582b625487dSandi        global $conf;
5830cecf9d5Sandi
584433bef32Sandi        $name = $this->_getLinkTitle($name, $url, $isImage);
5856f0c5dbfSandi
5866f0c5dbfSandi        // add protocol on simple short URLs
58726b8b699Sandi        if(substr($url,0,3) == 'ftp' && (substr($url,0,6) != 'ftp://')) $url = 'ftp://'.$url;
5886f0c5dbfSandi        if(substr($url,0,3) == 'www') $url = 'http://'.$url;
5890cecf9d5Sandi
5900cecf9d5Sandi        if ( !$isImage ) {
591b625487dSandi            $class='urlextern';
5920cecf9d5Sandi        } else {
593b625487dSandi            $class='media';
5940cecf9d5Sandi        }
5950cecf9d5Sandi
596b625487dSandi        //prepare for formating
597b625487dSandi        $link['target'] = $conf['target']['extern'];
598b625487dSandi        $link['style']  = '';
599b625487dSandi        $link['pre']    = '';
600b625487dSandi        $link['suf']    = '';
6015e163278SAndreas Gohr        $link['more']   = '';
602b625487dSandi        $link['class']  = $class;
603b625487dSandi        $link['url']    = $url;
604b625487dSandi        $link['name']   = $name;
605433bef32Sandi        $link['title']  = $this->_xmlEntities($url);
606b625487dSandi        if($conf['relnofollow']) $link['more'] .= ' rel="nofollow"';
6070cecf9d5Sandi
608b625487dSandi        //output formatted
609a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6100cecf9d5Sandi    }
6110cecf9d5Sandi
6120cecf9d5Sandi    /**
6130cecf9d5Sandi    */
61497a3e4e3Sandi    function interwikilink($match, $name = NULL, $wikiName, $wikiUri) {
615b625487dSandi        global $conf;
6160cecf9d5Sandi
61797a3e4e3Sandi        $link = array();
61897a3e4e3Sandi        $link['target'] = $conf['target']['interwiki'];
61997a3e4e3Sandi        $link['pre']    = '';
62097a3e4e3Sandi        $link['suf']    = '';
6215e163278SAndreas Gohr        $link['more']   = '';
622433bef32Sandi        $link['name']   = $this->_getLinkTitle($name, $wikiUri, $isImage);
6230cecf9d5Sandi
62497a3e4e3Sandi        //get interwiki URL
62597a3e4e3Sandi        if ( isset($this->interwiki[$wikiName]) ) {
62697a3e4e3Sandi            $url = $this->interwiki[$wikiName];
62797a3e4e3Sandi        } else {
62897a3e4e3Sandi            // Default to Google I'm feeling lucky
62997a3e4e3Sandi            $url = 'http://www.google.com/search?q={URL}&amp;btnI=lucky';
63097a3e4e3Sandi            $wikiName = 'go';
6310cecf9d5Sandi        }
6320cecf9d5Sandi
63397a3e4e3Sandi        if ( !$isImage ) {
6349d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$wikiName);
6359d2ddea4SAndreas Gohr            $link['class'] = "interwiki iw_$class";
6361c2d1019SAndreas Gohr        } else {
6371c2d1019SAndreas Gohr            $link['class'] = 'media';
63897a3e4e3Sandi        }
6390cecf9d5Sandi
64097a3e4e3Sandi        //do we stay at the same server? Use local target
64197a3e4e3Sandi        if( strpos($url,DOKU_URL) === 0 ){
64297a3e4e3Sandi            $link['target'] = $conf['target']['wiki'];
64397a3e4e3Sandi        }
6440cecf9d5Sandi
645b034216dSAndreas Gohr        //split into hash and url part
646b034216dSAndreas Gohr        list($wikiUri,$hash) = explode('#',$wikiUri,2);
647b034216dSAndreas Gohr
64897a3e4e3Sandi        //replace placeholder
64997a3e4e3Sandi        if(preg_match('#\{(URL|NAME|SCHEME|HOST|PORT|PATH|QUERY)\}#',$url)){
65097a3e4e3Sandi            //use placeholders
651b6c6979fSAndreas Gohr            $url = str_replace('{URL}',rawurlencode($wikiUri),$url);
65297a3e4e3Sandi            $url = str_replace('{NAME}',$wikiUri,$url);
65397a3e4e3Sandi            $parsed = parse_url($wikiUri);
65497a3e4e3Sandi            if(!$parsed['port']) $parsed['port'] = 80;
65597a3e4e3Sandi            $url = str_replace('{SCHEME}',$parsed['scheme'],$url);
65697a3e4e3Sandi            $url = str_replace('{HOST}',$parsed['host'],$url);
65797a3e4e3Sandi            $url = str_replace('{PORT}',$parsed['port'],$url);
65897a3e4e3Sandi            $url = str_replace('{PATH}',$parsed['path'],$url);
65997a3e4e3Sandi            $url = str_replace('{QUERY}',$parsed['query'],$url);
66097a3e4e3Sandi            $link['url'] = $url;
66197a3e4e3Sandi        }else{
66297a3e4e3Sandi            //default
663b6c6979fSAndreas Gohr            $link['url'] = $url.rawurlencode($wikiUri);
66497a3e4e3Sandi        }
665b6c6979fSAndreas Gohr        if($hash) $link['url'] .= '#'.rawurlencode($hash);
66697a3e4e3Sandi
66797a3e4e3Sandi        $link['title'] = htmlspecialchars($link['url']);
66897a3e4e3Sandi
66997a3e4e3Sandi        //output formatted
670a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6710cecf9d5Sandi    }
6720cecf9d5Sandi
6730cecf9d5Sandi    /**
6740cecf9d5Sandi     */
6751d47afe1Sandi    function windowssharelink($url, $name = NULL) {
6761d47afe1Sandi        global $conf;
6771d47afe1Sandi        global $lang;
6781d47afe1Sandi        //simple setup
6791d47afe1Sandi        $link['target'] = $conf['target']['windows'];
6801d47afe1Sandi        $link['pre']    = '';
6811d47afe1Sandi        $link['suf']   = '';
6821d47afe1Sandi        $link['style']  = '';
6831d47afe1Sandi        //Display error on browsers other than IE
6841d47afe1Sandi        $link['more'] = 'onclick="if(document.all == null){alert(\''.
685433bef32Sandi                        $this->_xmlEntities($lang['nosmblinks'],ENT_QUOTES).
6861d47afe1Sandi                        '\');}" onkeypress="if(document.all == null){alert(\''.
687433bef32Sandi                        $this->_xmlEntities($lang['nosmblinks'],ENT_QUOTES).'\');}"';
6880cecf9d5Sandi
689433bef32Sandi        $link['name'] = $this->_getLinkTitle($name, $url, $isImage);
6900cecf9d5Sandi        if ( !$isImage ) {
6911d47afe1Sandi            $link['class'] = 'windows';
6920cecf9d5Sandi        } else {
6931d47afe1Sandi            $link['class'] = 'media';
6940cecf9d5Sandi        }
6950cecf9d5Sandi
6960cecf9d5Sandi
697433bef32Sandi        $link['title'] = $this->_xmlEntities($url);
6981d47afe1Sandi        $url = str_replace('\\','/',$url);
6991d47afe1Sandi        $url = 'file:///'.$url;
7001d47afe1Sandi        $link['url'] = $url;
7010cecf9d5Sandi
7021d47afe1Sandi        //output formatted
703a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7040cecf9d5Sandi    }
7050cecf9d5Sandi
70671352defSandi    function emaillink($address, $name = NULL) {
70771352defSandi        global $conf;
70871352defSandi        //simple setup
70971352defSandi        $link = array();
71071352defSandi        $link['target'] = '';
71171352defSandi        $link['pre']    = '';
71271352defSandi        $link['suf']   = '';
71371352defSandi        $link['style']  = '';
71471352defSandi        $link['more']   = '';
7150cecf9d5Sandi
71671352defSandi        //we just test for image here - we need to encode the title our self
717433bef32Sandi        $this->_getLinkTitle($name, $address, $isImage);
7180cecf9d5Sandi        if ( !$isImage ) {
719776b36ecSAndreas Gohr            $link['class']='mail JSnocheck';
7200cecf9d5Sandi        } else {
721776b36ecSAndreas Gohr            $link['class']='media JSnocheck';
7220cecf9d5Sandi        }
7230cecf9d5Sandi
72407738714SAndreas Gohr        $address = $this->_xmlEntities($address);
72500a7b5adSEsther Brunner        $address = obfuscate($address);
72600a7b5adSEsther Brunner        $title   = $address;
72771352defSandi        if(empty($name)){
72800a7b5adSEsther Brunner            $name = $address;
72971352defSandi        }else{
730433bef32Sandi            $name = $this->_xmlEntities($name);
73171352defSandi        }
7320cecf9d5Sandi
733776b36ecSAndreas Gohr        if($conf['mailguard'] == 'visible') $address = rawurlencode($address);
734776b36ecSAndreas Gohr
735776b36ecSAndreas Gohr        $link['url']   = 'mailto:'.$address;
73671352defSandi        $link['name']  = $name;
73771352defSandi        $link['title'] = $title;
7380cecf9d5Sandi
73971352defSandi        //output formatted
740a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7410cecf9d5Sandi    }
7420cecf9d5Sandi
7434826ab45Sandi    function internalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
744dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
7453685f775Sandi        global $conf;
74637e34a5eSandi        global $ID;
74737e34a5eSandi        resolve_mediaid(getNS($ID),$src, $exists);
7480cecf9d5Sandi
7493685f775Sandi        $link = array();
7503685f775Sandi        $link['class']  = 'media';
7513685f775Sandi        $link['style']  = '';
7523685f775Sandi        $link['pre']    = '';
7533685f775Sandi        $link['suf']    = '';
7545e163278SAndreas Gohr        $link['more']   = '';
7553685f775Sandi        $link['target'] = $conf['target']['media'];
7563685f775Sandi
757433bef32Sandi        $link['title']  = $this->_xmlEntities($src);
75855efc227SAndreas Gohr        list($ext,$mime) = mimetype($src);
7595667eb65SAndreas Gohr        if(substr($mime,0,5) == 'image'){
760dc673a5bSjoe.lapp             $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),($linking=='direct'));
7612ca14335SEsther Brunner         }elseif($mime == 'application/x-shockwave-flash'){
7622ca14335SEsther Brunner             // don't link flash movies
7632ca14335SEsther Brunner             $noLink = TRUE;
76455efc227SAndreas Gohr         }else{
7652ca14335SEsther Brunner             // add file icons
7669d2ddea4SAndreas Gohr             $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
7679d2ddea4SAndreas Gohr             $link['class'] .= ' mediafile mf_'.$class;
7686de3759aSAndreas Gohr             $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),true);
76955efc227SAndreas Gohr         }
770433bef32Sandi         $link['name']   = $this->_media ($src, $title, $align, $width, $height, $cache);
7713685f775Sandi
7723685f775Sandi         //output formatted
773dc673a5bSjoe.lapp         if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
7742ca14335SEsther Brunner         else $this->doc .= $this->_formatLink($link);
7750cecf9d5Sandi    }
7760cecf9d5Sandi
7770cecf9d5Sandi    /**
7784826ab45Sandi     * @todo don't add link for flash
7790cecf9d5Sandi     */
7804826ab45Sandi    function externalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
781dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
7823685f775Sandi        global $conf;
7830cecf9d5Sandi
7843685f775Sandi        $link = array();
7853685f775Sandi        $link['class']  = 'media';
7863685f775Sandi        $link['style']  = '';
7873685f775Sandi        $link['pre']    = '';
7883685f775Sandi        $link['suf']    = '';
7895e163278SAndreas Gohr        $link['more']   = '';
7903685f775Sandi        $link['target'] = $conf['target']['media'];
7913685f775Sandi
792433bef32Sandi        $link['title']  = $this->_xmlEntities($src);
7936de3759aSAndreas Gohr        $link['url']    = ml($src,array('cache'=>$cache));
794433bef32Sandi        $link['name']   = $this->_media ($src, $title, $align, $width, $height, $cache);
7953685f775Sandi
7963685f775Sandi
7972ca14335SEsther Brunner        list($ext,$mime) = mimetype($src);
7982ca14335SEsther Brunner        if(substr($mime,0,5) == 'image'){
7992ca14335SEsther Brunner             // link only jpeg images
8002ca14335SEsther Brunner             // if ($ext != 'jpg' && $ext != 'jpeg') $noLink = TRUE;
8012ca14335SEsther Brunner        }elseif($mime == 'application/x-shockwave-flash'){
8022ca14335SEsther Brunner             // don't link flash movies
8032ca14335SEsther Brunner             $noLink = TRUE;
8042ca14335SEsther Brunner        }else{
8052ca14335SEsther Brunner             // add file icons
806d15166e5SAndreas Gohr             $link['class'] .= ' mediafile mf_'.$ext;
8072ca14335SEsther Brunner         }
8082ca14335SEsther Brunner
8093685f775Sandi        //output formatted
810dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
8112ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
8120cecf9d5Sandi    }
8130cecf9d5Sandi
8144826ab45Sandi    /**
8153fd0b676Sandi     * Renders an RSS feed using Magpie
816b625487dSandi     *
817b625487dSandi     * @author Andreas Gohr <andi@splitbrain.org>
818b625487dSandi     */
819b625487dSandi    function rss ($url){
820b625487dSandi        global $lang;
821b625487dSandi        define('MAGPIE_CACHE_ON', false); //we do our own caching
822b625487dSandi        define('MAGPIE_DIR', DOKU_INC.'inc/magpie/');
823b625487dSandi        define('MAGPIE_OUTPUT_ENCODING','UTF-8'); //return all feeds as UTF-8
824b625487dSandi        require_once(MAGPIE_DIR.'/rss_fetch.inc');
825b625487dSandi
826b625487dSandi        //disable warning while fetching
827b625487dSandi        $elvl = error_reporting(E_ERROR);
828b625487dSandi        $rss  = fetch_rss($url);
829b625487dSandi        error_reporting($elvl);
830b625487dSandi
831a2d649c4Sandi        $this->doc .= '<ul class="rss">';
832b625487dSandi        if($rss){
833b625487dSandi            foreach ($rss->items as $item ) {
834a2d649c4Sandi                $this->doc .= '<li>';
835b625487dSandi                $this->externallink($item['link'],$item['title']);
836a2d649c4Sandi                $this->doc .= '</li>';
837b625487dSandi            }
838b625487dSandi        }else{
839a2d649c4Sandi            $this->doc .= '<li>';
840a2d649c4Sandi            $this->doc .= '<em>'.$lang['rssfailed'].'</em>';
841b625487dSandi            $this->externallink($url);
842a2d649c4Sandi            $this->doc .= '</li>';
843b625487dSandi        }
844a2d649c4Sandi        $this->doc .= '</ul>';
845b625487dSandi    }
846b625487dSandi
8470cecf9d5Sandi    // $numrows not yet implemented
8480cecf9d5Sandi    function table_open($maxcols = NULL, $numrows = NULL){
849a2d649c4Sandi        $this->doc .= '<table class="inline">'.DOKU_LF;
8500cecf9d5Sandi    }
8510cecf9d5Sandi
8520cecf9d5Sandi    function table_close(){
853a2d649c4Sandi        $this->doc .= '</table>'.DOKU_LF.'<br />'.DOKU_LF;
8540cecf9d5Sandi    }
8550cecf9d5Sandi
8560cecf9d5Sandi    function tablerow_open(){
857a2d649c4Sandi        $this->doc .= DOKU_TAB . '<tr>' . DOKU_LF . DOKU_TAB . DOKU_TAB;
8580cecf9d5Sandi    }
8590cecf9d5Sandi
8600cecf9d5Sandi    function tablerow_close(){
861a2d649c4Sandi        $this->doc .= DOKU_LF . DOKU_TAB . '</tr>' . DOKU_LF;
8620cecf9d5Sandi    }
8630cecf9d5Sandi
8640cecf9d5Sandi    function tableheader_open($colspan = 1, $align = NULL){
865a2d649c4Sandi        $this->doc .= '<th';
8660cecf9d5Sandi        if ( !is_null($align) ) {
867a2d649c4Sandi            $this->doc .= ' class="'.$align.'align"';
8680cecf9d5Sandi        }
8690cecf9d5Sandi        if ( $colspan > 1 ) {
870a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8710cecf9d5Sandi        }
872a2d649c4Sandi        $this->doc .= '>';
8730cecf9d5Sandi    }
8740cecf9d5Sandi
8750cecf9d5Sandi    function tableheader_close(){
876a2d649c4Sandi        $this->doc .= '</th>';
8770cecf9d5Sandi    }
8780cecf9d5Sandi
8790cecf9d5Sandi    function tablecell_open($colspan = 1, $align = NULL){
880a2d649c4Sandi        $this->doc .= '<td';
8810cecf9d5Sandi        if ( !is_null($align) ) {
882a2d649c4Sandi            $this->doc .= ' class="'.$align.'align"';
8830cecf9d5Sandi        }
8840cecf9d5Sandi        if ( $colspan > 1 ) {
885a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8860cecf9d5Sandi        }
887a2d649c4Sandi        $this->doc .= '>';
8880cecf9d5Sandi    }
8890cecf9d5Sandi
8900cecf9d5Sandi    function tablecell_close(){
891a2d649c4Sandi        $this->doc .= '</td>';
8920cecf9d5Sandi    }
8930cecf9d5Sandi
8940cecf9d5Sandi    //----------------------------------------------------------
8950cecf9d5Sandi    // Utils
8960cecf9d5Sandi
897ba11bd29Sandi    /**
8983fd0b676Sandi     * Build a link
8993fd0b676Sandi     *
9003fd0b676Sandi     * Assembles all parts defined in $link returns HTML for the link
901ba11bd29Sandi     *
902ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
903ba11bd29Sandi     */
904433bef32Sandi    function _formatLink($link){
905ba11bd29Sandi        //make sure the url is XHTML compliant (skip mailto)
906ba11bd29Sandi        if(substr($link['url'],0,7) != 'mailto:'){
907ba11bd29Sandi            $link['url'] = str_replace('&','&amp;',$link['url']);
908ba11bd29Sandi            $link['url'] = str_replace('&amp;amp;','&amp;',$link['url']);
909ba11bd29Sandi        }
910ba11bd29Sandi        //remove double encodings in titles
911ba11bd29Sandi        $link['title'] = str_replace('&amp;amp;','&amp;',$link['title']);
912ba11bd29Sandi
913453493f2SAndreas Gohr        // be sure there are no bad chars in url or title
914453493f2SAndreas Gohr        // (we can't do this for name because it can contain an img tag)
915453493f2SAndreas Gohr        $link['url']   = strtr($link['url'],array('>'=>'%3E','<'=>'%3C','"'=>'%22'));
916453493f2SAndreas Gohr        $link['title'] = strtr($link['title'],array('>'=>'&gt;','<'=>'&lt;','"'=>'&quot;'));
917453493f2SAndreas Gohr
918ba11bd29Sandi        $ret  = '';
919ba11bd29Sandi        $ret .= $link['pre'];
920ba11bd29Sandi        $ret .= '<a href="'.$link['url'].'"';
921ba11bd29Sandi        if($link['class'])  $ret .= ' class="'.$link['class'].'"';
922ba11bd29Sandi        if($link['target']) $ret .= ' target="'.$link['target'].'"';
923ba11bd29Sandi        if($link['title'])  $ret .= ' title="'.$link['title'].'"';
924ba11bd29Sandi        if($link['style'])  $ret .= ' style="'.$link['style'].'"';
925ba11bd29Sandi        if($link['more'])   $ret .= ' '.$link['more'];
926ba11bd29Sandi        $ret .= '>';
927ba11bd29Sandi        $ret .= $link['name'];
928ba11bd29Sandi        $ret .= '</a>';
929ba11bd29Sandi        $ret .= $link['suf'];
930ba11bd29Sandi        return $ret;
931ba11bd29Sandi    }
932ba11bd29Sandi
933ba11bd29Sandi    /**
934ba11bd29Sandi     * Removes any Namespace from the given name but keeps
935ba11bd29Sandi     * casing and special chars
936ba11bd29Sandi     *
937ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
938ba11bd29Sandi     */
939433bef32Sandi    function _simpleTitle($name){
940ba11bd29Sandi        global $conf;
941bd111079Sandi
942ba11bd29Sandi        if($conf['useslash']){
943ba11bd29Sandi            $nssep = '[:;/]';
944ba11bd29Sandi        }else{
945ba11bd29Sandi            $nssep = '[:;]';
946ba11bd29Sandi        }
947bd111079Sandi        $name = preg_replace('!.*'.$nssep.'!','',$name);
948bd111079Sandi        //if there is a hash we use the ancor name only
949bd111079Sandi        $name = preg_replace('!.*#!','',$name);
950bd111079Sandi        return $name;
951ba11bd29Sandi    }
952ba11bd29Sandi
9533fd0b676Sandi    /**
9543fd0b676Sandi     * Renders internal and external media
9553fd0b676Sandi     *
9563fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
9573fd0b676Sandi     */
9583fd0b676Sandi    function _media ($src, $title=NULL, $align=NULL, $width=NULL,
9593fd0b676Sandi                      $height=NULL, $cache=NULL) {
9603fd0b676Sandi
9613fd0b676Sandi        $ret = '';
9623fd0b676Sandi
9633fd0b676Sandi        list($ext,$mime) = mimetype($src);
9643fd0b676Sandi        if(substr($mime,0,5) == 'image'){
9653fd0b676Sandi            //add image tag
9666de3759aSAndreas Gohr            $ret .= '<img src="'.ml($src,array('w'=>$width,'h'=>$height,'cache'=>$cache)).'"';
9673fd0b676Sandi            $ret .= ' class="media'.$align.'"';
9683fd0b676Sandi
9693fd0b676Sandi            if (!is_null($title)) {
9703fd0b676Sandi                $ret .= ' title="'.$this->_xmlEntities($title).'"';
9713fd0b676Sandi                $ret .= ' alt="'.$this->_xmlEntities($title).'"';
97255efc227SAndreas Gohr            }elseif($ext == 'jpg' || $ext == 'jpeg'){
97355efc227SAndreas Gohr                //try to use the caption from IPTC/EXIF
97455efc227SAndreas Gohr                require_once(DOKU_INC.'inc/JpegMeta.php');
97555efc227SAndreas Gohr                $jpeg =& new JpegMeta(mediaFN($src));
97655efc227SAndreas Gohr                if($jpeg !== false) $cap = $jpeg->getTitle();
97755efc227SAndreas Gohr                if($cap){
97855efc227SAndreas Gohr                    $ret .= ' title="'.$this->_xmlEntities($cap).'"';
97955efc227SAndreas Gohr                    $ret .= ' alt="'.$this->_xmlEntities($cap).'"';
98055efc227SAndreas Gohr                }
9813fd0b676Sandi            }else{
9823fd0b676Sandi                $ret .= ' alt=""';
9833fd0b676Sandi            }
9843fd0b676Sandi
9853fd0b676Sandi            if ( !is_null($width) )
9863fd0b676Sandi                $ret .= ' width="'.$this->_xmlEntities($width).'"';
9873fd0b676Sandi
9883fd0b676Sandi            if ( !is_null($height) )
9893fd0b676Sandi                $ret .= ' height="'.$this->_xmlEntities($height).'"';
9903fd0b676Sandi
9913fd0b676Sandi            $ret .= ' />';
9923fd0b676Sandi
9933fd0b676Sandi        }elseif($mime == 'application/x-shockwave-flash'){
9943fd0b676Sandi            $ret .= '<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"'.
9953fd0b676Sandi                    ' codebase="http://download.macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version=6,0,40,0"';
9963fd0b676Sandi            if ( !is_null($width) ) $ret .= ' width="'.$this->_xmlEntities($width).'"';
9973fd0b676Sandi            if ( !is_null($height) ) $ret .= ' height="'.$this->_xmlEntities($height).'"';
9983fd0b676Sandi            $ret .= '>'.DOKU_LF;
9996de3759aSAndreas Gohr            $ret .= '<param name="movie" value="'.ml($src).'" />'.DOKU_LF;
10003fd0b676Sandi            $ret .= '<param name="quality" value="high" />'.DOKU_LF;
10016de3759aSAndreas Gohr            $ret .= '<embed src="'.ml($src).'"'.
10023fd0b676Sandi                    ' quality="high"';
10033fd0b676Sandi            if ( !is_null($width) ) $ret .= ' width="'.$this->_xmlEntities($width).'"';
10043fd0b676Sandi            if ( !is_null($height) ) $ret .= ' height="'.$this->_xmlEntities($height).'"';
10053fd0b676Sandi            $ret .= ' type="application/x-shockwave-flash"'.
10063fd0b676Sandi                    ' pluginspage="http://www.macromedia.com/go/getflashplayer"></embed>'.DOKU_LF;
10073fd0b676Sandi            $ret .= '</object>'.DOKU_LF;
10083fd0b676Sandi
10093fd0b676Sandi        }elseif(!is_null($title)){
10103fd0b676Sandi            // well at least we have a title to display
10113fd0b676Sandi            $ret .= $this->_xmlEntities($title);
10123fd0b676Sandi        }else{
10135291ca3aSAndreas Gohr            // just show the sourcename
10145291ca3aSAndreas Gohr            $ret .= $this->_xmlEntities(noNS($src));
10153fd0b676Sandi        }
10163fd0b676Sandi
10173fd0b676Sandi        return $ret;
10183fd0b676Sandi    }
10193fd0b676Sandi
1020433bef32Sandi    function _xmlEntities($string) {
10210cecf9d5Sandi        return htmlspecialchars($string);
10220cecf9d5Sandi    }
10230cecf9d5Sandi
10248a831f2bSAndreas Gohr    /**
10258a831f2bSAndreas Gohr     * Creates a linkid from a headline
1026c5a8fd96SAndreas Gohr     *
1027c5a8fd96SAndreas Gohr     * @param string  $title   The headline title
1028c5a8fd96SAndreas Gohr     * @param boolean $create  Create a new unique ID?
1029c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
10308a831f2bSAndreas Gohr     */
1031c5a8fd96SAndreas Gohr    function _headerToLink($title,$create=false) {
1032c5a8fd96SAndreas Gohr        $title = str_replace(':','',cleanID($title,true)); //force ASCII
10338a831f2bSAndreas Gohr        $title = ltrim($title,'0123456789');
10348a831f2bSAndreas Gohr        if(empty($title)) $title='section';
1035c5a8fd96SAndreas Gohr
1036c5a8fd96SAndreas Gohr        if($create){
1037c5a8fd96SAndreas Gohr            // make sure tiles are unique
1038c5a8fd96SAndreas Gohr            $num = '';
1039c5a8fd96SAndreas Gohr            while(in_array($title.$num,$this->headers)){
1040c5a8fd96SAndreas Gohr                ($num) ? $num = 1 : $num++;
1041c5a8fd96SAndreas Gohr            }
1042c5a8fd96SAndreas Gohr            $title = $title.$num;
1043c5a8fd96SAndreas Gohr            $this->headers[] = $title;
1044c5a8fd96SAndreas Gohr        }
1045c5a8fd96SAndreas Gohr
10468a831f2bSAndreas Gohr        return $title;
10470cecf9d5Sandi    }
10480cecf9d5Sandi
1049af587fa8Sandi    /**
1050af587fa8Sandi     * Adds code for section editing button
10513fd0b676Sandi     *
10523fd0b676Sandi     * This is just aplaceholder and gets replace by the button if
10533fd0b676Sandi     * section editing is allowed
10543fd0b676Sandi     *
10553fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
1056af587fa8Sandi     */
1057433bef32Sandi    function _secedit($f, $t){
1058a2d649c4Sandi        $this->doc .= '<!-- SECTION ['.$f.'-'.$t.'] -->';
1059af587fa8Sandi    }
1060af587fa8Sandi
10613fd0b676Sandi    /**
10623fd0b676Sandi     * Construct a title and handle images in titles
10633fd0b676Sandi     *
10640b7c14c2Sandi     * @author Harry Fuecks <hfuecks@gmail.com>
10653fd0b676Sandi     */
1066433bef32Sandi    function _getLinkTitle($title, $default, & $isImage, $id=NULL) {
1067bb0a59d4Sjan        global $conf;
1068bb0a59d4Sjan
10690cecf9d5Sandi        $isImage = FALSE;
10700cecf9d5Sandi        if ( is_null($title) ) {
1071bb0a59d4Sjan            if ($conf['useheading'] && $id) {
1072bb0a59d4Sjan                $heading = p_get_first_heading($id);
1073bb0a59d4Sjan                if ($heading) {
1074433bef32Sandi                    return $this->_xmlEntities($heading);
1075bb0a59d4Sjan                }
1076bb0a59d4Sjan            }
1077433bef32Sandi            return $this->_xmlEntities($default);
10780cecf9d5Sandi        } else if ( is_string($title) ) {
1079433bef32Sandi            return $this->_xmlEntities($title);
10800cecf9d5Sandi        } else if ( is_array($title) ) {
10810cecf9d5Sandi            $isImage = TRUE;
1082433bef32Sandi            return $this->_imageTitle($title);
10830cecf9d5Sandi        }
10840cecf9d5Sandi    }
10850cecf9d5Sandi
10860cecf9d5Sandi    /**
10873fd0b676Sandi     * Returns an HTML code for images used in link titles
10883fd0b676Sandi     *
10893fd0b676Sandi     * @todo Resolve namespace on internal images
10903fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
10910cecf9d5Sandi     */
1092433bef32Sandi    function _imageTitle($img) {
1093433bef32Sandi        return $this->_media($img['src'],
10944826ab45Sandi                              $img['title'],
10954826ab45Sandi                              $img['align'],
10964826ab45Sandi                              $img['width'],
10974826ab45Sandi                              $img['height'],
10984826ab45Sandi                              $img['cache']);
10990cecf9d5Sandi    }
11000cecf9d5Sandi}
11010cecf9d5Sandi
11024826ab45Sandi//Setup VIM: ex: et ts=4 enc=utf-8 :
1103