xref: /dokuwiki/inc/parser/xhtml.php (revision 91459163e4ff1c28a910340507960898f4d8e126)
10cecf9d5Sandi<?php
2b625487dSandi/**
3b625487dSandi * Renderer for XHTML output
4b625487dSandi *
5b625487dSandi * @author Harry Fuecks <hfuecks@gmail.com>
6b625487dSandi * @author Andreas Gohr <andi@splitbrain.org>
7b625487dSandi */
8fa8adffeSAndreas Gohrif(!defined('DOKU_INC')) die('meh.');
90cecf9d5Sandi
100cecf9d5Sandiif ( !defined('DOKU_LF') ) {
110cecf9d5Sandi    // Some whitespace to help View > Source
120cecf9d5Sandi    define ('DOKU_LF',"\n");
130cecf9d5Sandi}
140cecf9d5Sandi
150cecf9d5Sandiif ( !defined('DOKU_TAB') ) {
160cecf9d5Sandi    // Some whitespace to help View > Source
170cecf9d5Sandi    define ('DOKU_TAB',"\t");
180cecf9d5Sandi}
190cecf9d5Sandi
200e1c636eSandirequire_once DOKU_INC . 'inc/parser/renderer.php';
2156fa9a3fSAndreas Gohrrequire_once DOKU_INC . 'inc/html.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    var $footnotes = array();
35*91459163SAnika Henke    var $pnid = 0;
36*91459163SAnika Henke    var $lastlevel = 0;
37*91459163SAnika Henke    var $node = array(0,0,0,0,0);
387764a90aSandi    var $store = '';
397764a90aSandi
40b5742cedSPierre Spring    var $_counter = array(); // used as global counter, introduced for table classes
41b5742cedSPierre Spring
425f70445dSAndreas Gohr    function getFormat(){
435f70445dSAndreas Gohr        return 'xhtml';
445f70445dSAndreas Gohr    }
455f70445dSAndreas Gohr
465f70445dSAndreas Gohr
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() {
540cecf9d5Sandi        if ( count ($this->footnotes) > 0 ) {
55a2d649c4Sandi            $this->doc .= '<div class="footnotes">'.DOKU_LF;
56d74aace9Schris
57d74aace9Schris            $id = 0;
580cecf9d5Sandi            foreach ( $this->footnotes as $footnote ) {
59d74aace9Schris                $id++;   // the number of the current footnote
60d74aace9Schris
61d74aace9Schris                // check its not a placeholder that indicates actual footnote text is elsewhere
62d74aace9Schris                if (substr($footnote, 0, 5) != "@@FNT") {
63d74aace9Schris
64d74aace9Schris                    // open the footnote and set the anchor and backlink
65d74aace9Schris                    $this->doc .= '<div class="fn">';
6629bfcd16SAndreas Gohr                    $this->doc .= '<sup><a href="#fnt__'.$id.'" id="fn__'.$id.'" name="fn__'.$id.'" class="fn_bot">';
6729bfcd16SAndreas Gohr                    $this->doc .= $id.')</a></sup> '.DOKU_LF;
68d74aace9Schris
69d74aace9Schris                    // get any other footnotes that use the same markup
70d74aace9Schris                    $alt = array_keys($this->footnotes, "@@FNT$id");
71d74aace9Schris
72d74aace9Schris                    if (count($alt)) {
73d74aace9Schris                      foreach ($alt as $ref) {
74d74aace9Schris                        // set anchor and backlink for the other footnotes
7529bfcd16SAndreas Gohr                        $this->doc .= ', <sup><a href="#fnt__'.($ref+1).'" id="fn__'.($ref+1).'" name="fn__'.($ref+1).'" class="fn_bot">';
7629bfcd16SAndreas Gohr                        $this->doc .= ($ref+1).')</a></sup> '.DOKU_LF;
77d74aace9Schris                      }
78d74aace9Schris                    }
79d74aace9Schris
80d74aace9Schris                    // add footnote markup and close this footnote
81a2d649c4Sandi                    $this->doc .= $footnote;
82d74aace9Schris                    $this->doc .= '</div>' . DOKU_LF;
83d74aace9Schris                }
840cecf9d5Sandi            }
85a2d649c4Sandi            $this->doc .= '</div>'.DOKU_LF;
860cecf9d5Sandi        }
87c5a8fd96SAndreas Gohr
88b8595a66SAndreas Gohr        // Prepare the TOC
89851f2e89SAnika Henke        global $conf;
90851f2e89SAnika Henke        if($this->info['toc'] && is_array($this->toc) && $conf['tocminheads'] && count($this->toc) >= $conf['tocminheads']){
91b8595a66SAndreas Gohr            global $TOC;
92b8595a66SAndreas Gohr            $TOC = $this->toc;
930cecf9d5Sandi        }
943e55d035SAndreas Gohr
953e55d035SAndreas Gohr        // make sure there are no empty paragraphs
9627918226Schris        $this->doc = preg_replace('#<p>\s*</p>#','',$this->doc);
97*91459163SAnika Henke        if ($conf['purplenumbers']) $this->doc = preg_replace('#<p[^>]*>\s*<!--PN-->.*?(?:</p>)#','',$this->doc);
98e41c4da9SAndreas Gohr    }
990cecf9d5Sandi
100e7856beaSchris    function toc_additem($id, $text, $level) {
101af587fa8Sandi        global $conf;
102af587fa8Sandi
103c5a8fd96SAndreas Gohr        //handle TOC
104c5a8fd96SAndreas Gohr        if($level >= $conf['toptoclevel'] && $level <= $conf['maxtoclevel']){
1057d91652aSAndreas Gohr            $this->toc[] = html_mktocitem($id, $text, $level-$conf['toptoclevel']+1);
106c5a8fd96SAndreas Gohr        }
107e7856beaSchris    }
108e7856beaSchris
109e7856beaSchris    function header($text, $level, $pos) {
110*91459163SAnika Henke        global $conf;
111*91459163SAnika Henke        global $lang;
112bdd8111bSAndreas Gohr        if(!$text) return; //skip empty headlines
113e7856beaSchris
114e7856beaSchris        $hid = $this->_headerToLink($text,true);
115e7856beaSchris
116e7856beaSchris        //only add items within configured levels
117e7856beaSchris        $this->toc_additem($hid, $text, $level);
118c5a8fd96SAndreas Gohr
119*91459163SAnika Henke        // adjust $node to reflect hierarchy of levels
120*91459163SAnika Henke        $this->node[$level-1]++;
121*91459163SAnika Henke        if ($level < $this->lastlevel) {
122*91459163SAnika Henke            for ($i = 0; $i < $this->lastlevel-$level; $i++) {
123*91459163SAnika Henke                $this->node[$this->lastlevel-$i-1] = 0;
124*91459163SAnika Henke            }
125*91459163SAnika Henke        }
126*91459163SAnika Henke        $this->pnid = 0;
127*91459163SAnika Henke        $this->lastlevel = $level;
128*91459163SAnika Henke
129c5a8fd96SAndreas Gohr        // write the header
130c5a8fd96SAndreas Gohr        $this->doc .= DOKU_LF.'<h'.$level.'><a name="'.$hid.'" id="'.$hid.'">';
131a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
132*91459163SAnika Henke        $this->doc .= "</a>";
133*91459163SAnika Henke        if ($conf['purplenumbers']) $this->doc .= " <a href='#$hid' class='pn' title='".$lang['sectionlink']."'>§</a>";
134*91459163SAnika Henke        $this->doc .= "</h$level>".DOKU_LF;
1350cecf9d5Sandi    }
1360cecf9d5Sandi
13735dae8b0SBen Coburn     /**
13835dae8b0SBen Coburn     * Section edit marker is replaced by an edit button when
13935dae8b0SBen Coburn     * the page is editable. Replacement done in 'inc/html.php#html_secedit'
14035dae8b0SBen Coburn     *
14135dae8b0SBen Coburn     * @author Andreas Gohr <andi@splitbrain.org>
14235dae8b0SBen Coburn     * @author Ben Coburn   <btcoburn@silicodon.net>
14335dae8b0SBen Coburn     */
14435dae8b0SBen Coburn    function section_edit($start, $end, $level, $name) {
14535dae8b0SBen Coburn        global $conf;
14635dae8b0SBen Coburn
14735dae8b0SBen Coburn        if ($start!=-1 && $level<=$conf['maxseclevel']) {
14835dae8b0SBen Coburn            $name = str_replace('"', '', $name);
14935dae8b0SBen Coburn            $this->doc .= '<!-- SECTION "'.$name.'" ['.$start.'-'.(($end===0)?'':$end).'] -->';
15035dae8b0SBen Coburn        }
15135dae8b0SBen Coburn    }
15235dae8b0SBen Coburn
1530cecf9d5Sandi    function section_open($level) {
154a2d649c4Sandi        $this->doc .= "<div class=\"level$level\">".DOKU_LF;
1550cecf9d5Sandi    }
1560cecf9d5Sandi
1570cecf9d5Sandi    function section_close() {
158a2d649c4Sandi        $this->doc .= DOKU_LF.'</div>'.DOKU_LF;
1590cecf9d5Sandi    }
1600cecf9d5Sandi
1610cecf9d5Sandi    function cdata($text) {
162a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
1630cecf9d5Sandi    }
1640cecf9d5Sandi
1650cecf9d5Sandi    function p_open() {
166*91459163SAnika Henke        $this->doc .= DOKU_LF.'<p'.$this->_getPurpleNumberID().'>'.DOKU_LF;
1670cecf9d5Sandi    }
1680cecf9d5Sandi
1690cecf9d5Sandi    function p_close() {
170*91459163SAnika Henke        $this->doc .= DOKU_LF.$this->_getPurpleNumberLink().'</p>'.DOKU_LF;
1710cecf9d5Sandi    }
1720cecf9d5Sandi
1730cecf9d5Sandi    function linebreak() {
174a2d649c4Sandi        $this->doc .= '<br/>'.DOKU_LF;
1750cecf9d5Sandi    }
1760cecf9d5Sandi
1770cecf9d5Sandi    function hr() {
1784beabca9SAnika Henke        $this->doc .= '<hr />'.DOKU_LF;
1790cecf9d5Sandi    }
1800cecf9d5Sandi
1810cecf9d5Sandi    function strong_open() {
182a2d649c4Sandi        $this->doc .= '<strong>';
1830cecf9d5Sandi    }
1840cecf9d5Sandi
1850cecf9d5Sandi    function strong_close() {
186a2d649c4Sandi        $this->doc .= '</strong>';
1870cecf9d5Sandi    }
1880cecf9d5Sandi
1890cecf9d5Sandi    function emphasis_open() {
190a2d649c4Sandi        $this->doc .= '<em>';
1910cecf9d5Sandi    }
1920cecf9d5Sandi
1930cecf9d5Sandi    function emphasis_close() {
194a2d649c4Sandi        $this->doc .= '</em>';
1950cecf9d5Sandi    }
1960cecf9d5Sandi
1970cecf9d5Sandi    function underline_open() {
19802e51121SAnika Henke        $this->doc .= '<em class="u">';
1990cecf9d5Sandi    }
2000cecf9d5Sandi
2010cecf9d5Sandi    function underline_close() {
20202e51121SAnika Henke        $this->doc .= '</em>';
2030cecf9d5Sandi    }
2040cecf9d5Sandi
2050cecf9d5Sandi    function monospace_open() {
206a2d649c4Sandi        $this->doc .= '<code>';
2070cecf9d5Sandi    }
2080cecf9d5Sandi
2090cecf9d5Sandi    function monospace_close() {
210a2d649c4Sandi        $this->doc .= '</code>';
2110cecf9d5Sandi    }
2120cecf9d5Sandi
2130cecf9d5Sandi    function subscript_open() {
214a2d649c4Sandi        $this->doc .= '<sub>';
2150cecf9d5Sandi    }
2160cecf9d5Sandi
2170cecf9d5Sandi    function subscript_close() {
218a2d649c4Sandi        $this->doc .= '</sub>';
2190cecf9d5Sandi    }
2200cecf9d5Sandi
2210cecf9d5Sandi    function superscript_open() {
222a2d649c4Sandi        $this->doc .= '<sup>';
2230cecf9d5Sandi    }
2240cecf9d5Sandi
2250cecf9d5Sandi    function superscript_close() {
226a2d649c4Sandi        $this->doc .= '</sup>';
2270cecf9d5Sandi    }
2280cecf9d5Sandi
2290cecf9d5Sandi    function deleted_open() {
230a2d649c4Sandi        $this->doc .= '<del>';
2310cecf9d5Sandi    }
2320cecf9d5Sandi
2330cecf9d5Sandi    function deleted_close() {
234a2d649c4Sandi        $this->doc .= '</del>';
2350cecf9d5Sandi    }
2360cecf9d5Sandi
2373fd0b676Sandi    /**
2383fd0b676Sandi     * Callback for footnote start syntax
2393fd0b676Sandi     *
2403fd0b676Sandi     * All following content will go to the footnote instead of
241d74aace9Schris     * the document. To achieve this the previous rendered content
2423fd0b676Sandi     * is moved to $store and $doc is cleared
2433fd0b676Sandi     *
2443fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
2453fd0b676Sandi     */
2460cecf9d5Sandi    function footnote_open() {
2477764a90aSandi
2487764a90aSandi        // move current content to store and record footnote
2497764a90aSandi        $this->store = $this->doc;
2507764a90aSandi        $this->doc   = '';
2510cecf9d5Sandi    }
2520cecf9d5Sandi
2533fd0b676Sandi    /**
2543fd0b676Sandi     * Callback for footnote end syntax
2553fd0b676Sandi     *
2563fd0b676Sandi     * All rendered content is moved to the $footnotes array and the old
2573fd0b676Sandi     * content is restored from $store again
2583fd0b676Sandi     *
2593fd0b676Sandi     * @author Andreas Gohr
2603fd0b676Sandi     */
2610cecf9d5Sandi    function footnote_close() {
2627764a90aSandi
263d74aace9Schris        // recover footnote into the stack and restore old content
264d74aace9Schris        $footnote = $this->doc;
2657764a90aSandi        $this->doc = $this->store;
2667764a90aSandi        $this->store = '';
267d74aace9Schris
268d74aace9Schris        // check to see if this footnote has been seen before
269d74aace9Schris        $i = array_search($footnote, $this->footnotes);
270d74aace9Schris
271d74aace9Schris        if ($i === false) {
272d74aace9Schris            // its a new footnote, add it to the $footnotes array
273d74aace9Schris            $id = count($this->footnotes)+1;
274d74aace9Schris            $this->footnotes[count($this->footnotes)] = $footnote;
275d74aace9Schris        } else {
276d74aace9Schris            // seen this one before, translate the index to an id and save a placeholder
277d74aace9Schris            $i++;
278d74aace9Schris            $id = count($this->footnotes)+1;
279d74aace9Schris            $this->footnotes[count($this->footnotes)] = "@@FNT".($i);
280d74aace9Schris        }
281d74aace9Schris
2826b379cbfSAndreas Gohr        // output the footnote reference and link
28329bfcd16SAndreas Gohr        $this->doc .= '<sup><a href="#fn__'.$id.'" name="fnt__'.$id.'" id="fnt__'.$id.'" class="fn_top">'.$id.')</a></sup>';
2840cecf9d5Sandi    }
2850cecf9d5Sandi
2860cecf9d5Sandi    function listu_open() {
287a2d649c4Sandi        $this->doc .= '<ul>'.DOKU_LF;
2880cecf9d5Sandi    }
2890cecf9d5Sandi
2900cecf9d5Sandi    function listu_close() {
291a2d649c4Sandi        $this->doc .= '</ul>'.DOKU_LF;
2920cecf9d5Sandi    }
2930cecf9d5Sandi
2940cecf9d5Sandi    function listo_open() {
295a2d649c4Sandi        $this->doc .= '<ol>'.DOKU_LF;
2960cecf9d5Sandi    }
2970cecf9d5Sandi
2980cecf9d5Sandi    function listo_close() {
299a2d649c4Sandi        $this->doc .= '</ol>'.DOKU_LF;
3000cecf9d5Sandi    }
3010cecf9d5Sandi
3020cecf9d5Sandi    function listitem_open($level) {
303*91459163SAnika Henke        $this->doc .= '<li class="level'.$level.'"'.$this->_getPurpleNumberID().'>';
3040cecf9d5Sandi    }
3050cecf9d5Sandi
3060cecf9d5Sandi    function listitem_close() {
307a2d649c4Sandi        $this->doc .= '</li>'.DOKU_LF;
3080cecf9d5Sandi    }
3090cecf9d5Sandi
3100cecf9d5Sandi    function listcontent_open() {
31190db23d7Schris        $this->doc .= '<div class="li">';
3120cecf9d5Sandi    }
3130cecf9d5Sandi
3140cecf9d5Sandi    function listcontent_close() {
315*91459163SAnika Henke        $this->doc .= $this->_getPurpleNumberLink().'</div>'.DOKU_LF;
3160cecf9d5Sandi    }
3170cecf9d5Sandi
3180cecf9d5Sandi    function unformatted($text) {
319a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
3200cecf9d5Sandi    }
3210cecf9d5Sandi
3220cecf9d5Sandi    /**
3233fd0b676Sandi     * Execute PHP code if allowed
3243fd0b676Sandi     *
3255d568b99SChris Smith     * @param  string   $wrapper   html element to wrap result if $conf['phpok'] is okff
3265d568b99SChris Smith     *
3273fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3280cecf9d5Sandi     */
3295d568b99SChris Smith    function php($text, $wrapper='code') {
33035a56260SChris Smith        global $conf;
33135a56260SChris Smith
332d86d5af0SChris Smith        if($conf['phpok']){
333bad0b545Sandi          ob_start();
3344de671bcSandi          eval($text);
3353fd0b676Sandi          $this->doc .= ob_get_contents();
336bad0b545Sandi          ob_end_clean();
337d86d5af0SChris Smith        } else {
3385d568b99SChris Smith          $this->doc .= p_xhtml_cached_geshi($text, 'php', $wrapper);
339d86d5af0SChris Smith        }
3400cecf9d5Sandi    }
3410cecf9d5Sandi
34207f89c3cSAnika Henke    function phpblock($text) {
3435d568b99SChris Smith        $this->php($text, 'pre');
34407f89c3cSAnika Henke    }
34507f89c3cSAnika Henke
3460cecf9d5Sandi    /**
3473fd0b676Sandi     * Insert HTML if allowed
3483fd0b676Sandi     *
3495d568b99SChris Smith     * @param  string   $wrapper   html element to wrap result if $conf['htmlok'] is okff
3505d568b99SChris Smith     *
3513fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3520cecf9d5Sandi     */
3535d568b99SChris Smith    function html($text, $wrapper='code') {
35435a56260SChris Smith        global $conf;
35535a56260SChris Smith
356d86d5af0SChris Smith        if($conf['htmlok']){
357a2d649c4Sandi          $this->doc .= $text;
358d86d5af0SChris Smith        } else {
3595d568b99SChris Smith          $this->doc .= p_xhtml_cached_geshi($text, 'html4strict', $wrapper);
360d86d5af0SChris Smith        }
3614de671bcSandi    }
3620cecf9d5Sandi
36307f89c3cSAnika Henke    function htmlblock($text) {
3645d568b99SChris Smith        $this->html($text, 'pre');
36507f89c3cSAnika Henke    }
36607f89c3cSAnika Henke
3670cecf9d5Sandi    function preformatted($text) {
368*91459163SAnika Henke        $this->doc .= '<pre class="code"'.$this->_getPurpleNumberID().'>' . trim($this->_xmlEntities($text)) . $this->_getPurpleNumberLink(). '</pre>'. DOKU_LF;
3690cecf9d5Sandi    }
3700cecf9d5Sandi
3710cecf9d5Sandi    function file($text) {
372*91459163SAnika Henke        $this->doc .= '<pre class="file"'.$this->_getPurpleNumberID().'>' . trim($this->_xmlEntities($text)). $this->_getPurpleNumberLink(). '</pre>'. DOKU_LF;
3730cecf9d5Sandi    }
3740cecf9d5Sandi
3750cecf9d5Sandi    function quote_open() {
37696331712SAnika Henke        $this->doc .= '<blockquote><div class="no">'.DOKU_LF;
3770cecf9d5Sandi    }
3780cecf9d5Sandi
3790cecf9d5Sandi    function quote_close() {
38096331712SAnika Henke        $this->doc .= '</div></blockquote>'.DOKU_LF;
3810cecf9d5Sandi    }
3820cecf9d5Sandi
3830cecf9d5Sandi    /**
3843fd0b676Sandi     * Callback for code text
3853fd0b676Sandi     *
3863fd0b676Sandi     * Uses GeSHi to highlight language syntax
3873fd0b676Sandi     *
3883fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3890cecf9d5Sandi     */
3900cecf9d5Sandi    function code($text, $language = NULL) {
3914de671bcSandi        global $conf;
3920cecf9d5Sandi
3930cecf9d5Sandi        if ( is_null($language) ) {
3940cecf9d5Sandi            $this->preformatted($text);
3950cecf9d5Sandi        } else {
3968f7d700cSchris            $this->doc .= p_xhtml_cached_geshi($text, $language);
3970cecf9d5Sandi        }
3980cecf9d5Sandi    }
3990cecf9d5Sandi
4000cecf9d5Sandi    function acronym($acronym) {
4010cecf9d5Sandi
4020cecf9d5Sandi        if ( array_key_exists($acronym, $this->acronyms) ) {
4030cecf9d5Sandi
404433bef32Sandi            $title = $this->_xmlEntities($this->acronyms[$acronym]);
4050cecf9d5Sandi
406a2d649c4Sandi            $this->doc .= '<acronym title="'.$title
407433bef32Sandi                .'">'.$this->_xmlEntities($acronym).'</acronym>';
4080cecf9d5Sandi
4090cecf9d5Sandi        } else {
410a2d649c4Sandi            $this->doc .= $this->_xmlEntities($acronym);
4110cecf9d5Sandi        }
4120cecf9d5Sandi    }
4130cecf9d5Sandi
4140cecf9d5Sandi    function smiley($smiley) {
4150cecf9d5Sandi        if ( array_key_exists($smiley, $this->smileys) ) {
416433bef32Sandi            $title = $this->_xmlEntities($this->smileys[$smiley]);
417f62ea8a1Sandi            $this->doc .= '<img src="'.DOKU_BASE.'lib/images/smileys/'.$this->smileys[$smiley].
4184beabca9SAnika Henke                '" class="middle" alt="'.
419433bef32Sandi                    $this->_xmlEntities($smiley).'" />';
4200cecf9d5Sandi        } else {
421a2d649c4Sandi            $this->doc .= $this->_xmlEntities($smiley);
4220cecf9d5Sandi        }
4230cecf9d5Sandi    }
4240cecf9d5Sandi
425f62ea8a1Sandi    /*
4264de671bcSandi    * not used
4270cecf9d5Sandi    function wordblock($word) {
4280cecf9d5Sandi        if ( array_key_exists($word, $this->badwords) ) {
429a2d649c4Sandi            $this->doc .= '** BLEEP **';
4300cecf9d5Sandi        } else {
431a2d649c4Sandi            $this->doc .= $this->_xmlEntities($word);
4320cecf9d5Sandi        }
4330cecf9d5Sandi    }
4344de671bcSandi    */
4350cecf9d5Sandi
4360cecf9d5Sandi    function entity($entity) {
4370cecf9d5Sandi        if ( array_key_exists($entity, $this->entities) ) {
438a2d649c4Sandi            $this->doc .= $this->entities[$entity];
4390cecf9d5Sandi        } else {
440a2d649c4Sandi            $this->doc .= $this->_xmlEntities($entity);
4410cecf9d5Sandi        }
4420cecf9d5Sandi    }
4430cecf9d5Sandi
4440cecf9d5Sandi    function multiplyentity($x, $y) {
445a2d649c4Sandi        $this->doc .= "$x&times;$y";
4460cecf9d5Sandi    }
4470cecf9d5Sandi
4480cecf9d5Sandi    function singlequoteopening() {
44971b40da2SAnika Henke        global $lang;
45071b40da2SAnika Henke        $this->doc .= $lang['singlequoteopening'];
4510cecf9d5Sandi    }
4520cecf9d5Sandi
4530cecf9d5Sandi    function singlequoteclosing() {
45471b40da2SAnika Henke        global $lang;
45571b40da2SAnika Henke        $this->doc .= $lang['singlequoteclosing'];
4560cecf9d5Sandi    }
4570cecf9d5Sandi
45857d757d1SAndreas Gohr    function apostrophe() {
45957d757d1SAndreas Gohr        global $lang;
460a8bd192aSAndreas Gohr        $this->doc .= $lang['apostrophe'];
46157d757d1SAndreas Gohr    }
46257d757d1SAndreas Gohr
4630cecf9d5Sandi    function doublequoteopening() {
46471b40da2SAnika Henke        global $lang;
46571b40da2SAnika Henke        $this->doc .= $lang['doublequoteopening'];
4660cecf9d5Sandi    }
4670cecf9d5Sandi
4680cecf9d5Sandi    function doublequoteclosing() {
46971b40da2SAnika Henke        global $lang;
47071b40da2SAnika Henke        $this->doc .= $lang['doublequoteclosing'];
4710cecf9d5Sandi    }
4720cecf9d5Sandi
4730cecf9d5Sandi    /**
4740cecf9d5Sandi    */
4750cecf9d5Sandi    function camelcaselink($link) {
47611d0aa47Sandi      $this->internallink($link,$link);
4770cecf9d5Sandi    }
4780cecf9d5Sandi
4790b7c14c2Sandi
4800b7c14c2Sandi    function locallink($hash, $name = NULL){
4810b7c14c2Sandi        global $ID;
4820b7c14c2Sandi        $name  = $this->_getLinkTitle($name, $hash, $isImage);
4830b7c14c2Sandi        $hash  = $this->_headerToLink($hash);
4840b7c14c2Sandi        $title = $ID.' &crarr;';
4850b7c14c2Sandi        $this->doc .= '<a href="#'.$hash.'" title="'.$title.'" class="wikilink1">';
4860b7c14c2Sandi        $this->doc .= $name;
4870b7c14c2Sandi        $this->doc .= '</a>';
4880b7c14c2Sandi    }
4890b7c14c2Sandi
490cffcc403Sandi    /**
4913fd0b676Sandi     * Render an internal Wiki Link
4923fd0b676Sandi     *
493fe9ec250SChris Smith     * $search,$returnonly & $linktype are not for the renderer but are used
494cffcc403Sandi     * elsewhere - no need to implement them in other renderers
4953fd0b676Sandi     *
4963fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
497cffcc403Sandi     */
498fe9ec250SChris Smith    function internallink($id, $name = NULL, $search=NULL,$returnonly=false,$linktype='content') {
499ba11bd29Sandi        global $conf;
50037e34a5eSandi        global $ID;
5010339c872Sjan        // default name is based on $id as given
5020339c872Sjan        $default = $this->_simpleTitle($id);
503ad32e47eSAndreas Gohr
5040339c872Sjan        // now first resolve and clean up the $id
50537e34a5eSandi        resolve_pageid(getNS($ID),$id,$exists);
506fe9ec250SChris Smith        $name = $this->_getLinkTitle($name, $default, $isImage, $id, $linktype);
5070e1c636eSandi        if ( !$isImage ) {
5080e1c636eSandi            if ( $exists ) {
509ba11bd29Sandi                $class='wikilink1';
5100cecf9d5Sandi            } else {
511ba11bd29Sandi                $class='wikilink2';
51244a6b4c7SAndreas Gohr                $link['rel']='nofollow';
5130cecf9d5Sandi            }
5140cecf9d5Sandi        } else {
515ba11bd29Sandi            $class='media';
5160cecf9d5Sandi        }
5170cecf9d5Sandi
518a1685bedSandi        //keep hash anchor
519ce6b63d9Schris        list($id,$hash) = explode('#',$id,2);
520943dedc6SAndreas Gohr        if(!empty($hash)) $hash = $this->_headerToLink($hash);
521a1685bedSandi
522ba11bd29Sandi        //prepare for formating
523ba11bd29Sandi        $link['target'] = $conf['target']['wiki'];
524ba11bd29Sandi        $link['style']  = '';
525ba11bd29Sandi        $link['pre']    = '';
526ba11bd29Sandi        $link['suf']    = '';
52740eb54bbSjan        // highlight link to current page
52840eb54bbSjan        if ($id == $ID) {
52992795d04Sandi            $link['pre']    = '<span class="curid">';
53092795d04Sandi            $link['suf']    = '</span>';
53140eb54bbSjan        }
5325e163278SAndreas Gohr        $link['more']   = '';
533ba11bd29Sandi        $link['class']  = $class;
534ba11bd29Sandi        $link['url']    = wl($id);
535ba11bd29Sandi        $link['name']   = $name;
536ba11bd29Sandi        $link['title']  = $id;
537723d78dbSandi        //add search string
538723d78dbSandi        if($search){
539546d3a99SAndreas Gohr            ($conf['userewrite']) ? $link['url'].='?' : $link['url'].='&amp;';
540546d3a99SAndreas Gohr            if(is_array($search)){
541546d3a99SAndreas Gohr                $search = array_map('rawurlencode',$search);
542546d3a99SAndreas Gohr                $link['url'] .= 's[]='.join('&amp;s[]=',$search);
543546d3a99SAndreas Gohr            }else{
544546d3a99SAndreas Gohr                $link['url'] .= 's='.rawurlencode($search);
545546d3a99SAndreas Gohr            }
546723d78dbSandi        }
547723d78dbSandi
548a1685bedSandi        //keep hash
549a1685bedSandi        if($hash) $link['url'].='#'.$hash;
550a1685bedSandi
551ba11bd29Sandi        //output formatted
552cffcc403Sandi        if($returnonly){
553cffcc403Sandi            return $this->_formatLink($link);
554cffcc403Sandi        }else{
555a2d649c4Sandi            $this->doc .= $this->_formatLink($link);
5560cecf9d5Sandi        }
557cffcc403Sandi    }
5580cecf9d5Sandi
559b625487dSandi    function externallink($url, $name = NULL) {
560b625487dSandi        global $conf;
5610cecf9d5Sandi
562433bef32Sandi        $name = $this->_getLinkTitle($name, $url, $isImage);
5636f0c5dbfSandi
5640cecf9d5Sandi        if ( !$isImage ) {
565b625487dSandi            $class='urlextern';
5660cecf9d5Sandi        } else {
567b625487dSandi            $class='media';
5680cecf9d5Sandi        }
5690cecf9d5Sandi
570b625487dSandi        //prepare for formating
571b625487dSandi        $link['target'] = $conf['target']['extern'];
572b625487dSandi        $link['style']  = '';
573b625487dSandi        $link['pre']    = '';
574b625487dSandi        $link['suf']    = '';
5755e163278SAndreas Gohr        $link['more']   = '';
576b625487dSandi        $link['class']  = $class;
577b625487dSandi        $link['url']    = $url;
578e1c10e4dSchris
579b625487dSandi        $link['name']   = $name;
580433bef32Sandi        $link['title']  = $this->_xmlEntities($url);
581b625487dSandi        if($conf['relnofollow']) $link['more'] .= ' rel="nofollow"';
5820cecf9d5Sandi
583b625487dSandi        //output formatted
584a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
5850cecf9d5Sandi    }
5860cecf9d5Sandi
5870cecf9d5Sandi    /**
5880cecf9d5Sandi    */
58997a3e4e3Sandi    function interwikilink($match, $name = NULL, $wikiName, $wikiUri) {
590b625487dSandi        global $conf;
5910cecf9d5Sandi
59297a3e4e3Sandi        $link = array();
59397a3e4e3Sandi        $link['target'] = $conf['target']['interwiki'];
59497a3e4e3Sandi        $link['pre']    = '';
59597a3e4e3Sandi        $link['suf']    = '';
5965e163278SAndreas Gohr        $link['more']   = '';
597433bef32Sandi        $link['name']   = $this->_getLinkTitle($name, $wikiUri, $isImage);
5980cecf9d5Sandi
59997a3e4e3Sandi        //get interwiki URL
6001f82fabeSAndreas Gohr        $url = $this->_resolveInterWiki($wikiName,$wikiUri);
6010cecf9d5Sandi
60297a3e4e3Sandi        if ( !$isImage ) {
6039d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$wikiName);
6049d2ddea4SAndreas Gohr            $link['class'] = "interwiki iw_$class";
6051c2d1019SAndreas Gohr        } else {
6061c2d1019SAndreas Gohr            $link['class'] = 'media';
60797a3e4e3Sandi        }
6080cecf9d5Sandi
60997a3e4e3Sandi        //do we stay at the same server? Use local target
61097a3e4e3Sandi        if( strpos($url,DOKU_URL) === 0 ){
61197a3e4e3Sandi            $link['target'] = $conf['target']['wiki'];
61297a3e4e3Sandi        }
6130cecf9d5Sandi
61497a3e4e3Sandi        $link['url'] = $url;
61597a3e4e3Sandi        $link['title'] = htmlspecialchars($link['url']);
61697a3e4e3Sandi
61797a3e4e3Sandi        //output formatted
618a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6190cecf9d5Sandi    }
6200cecf9d5Sandi
6210cecf9d5Sandi    /**
6220cecf9d5Sandi     */
6231d47afe1Sandi    function windowssharelink($url, $name = NULL) {
6241d47afe1Sandi        global $conf;
6251d47afe1Sandi        global $lang;
6261d47afe1Sandi        //simple setup
6271d47afe1Sandi        $link['target'] = $conf['target']['windows'];
6281d47afe1Sandi        $link['pre']    = '';
6291d47afe1Sandi        $link['suf']   = '';
6301d47afe1Sandi        $link['style']  = '';
6310cecf9d5Sandi
632433bef32Sandi        $link['name'] = $this->_getLinkTitle($name, $url, $isImage);
6330cecf9d5Sandi        if ( !$isImage ) {
6341d47afe1Sandi            $link['class'] = 'windows';
6350cecf9d5Sandi        } else {
6361d47afe1Sandi            $link['class'] = 'media';
6370cecf9d5Sandi        }
6380cecf9d5Sandi
6390cecf9d5Sandi
640433bef32Sandi        $link['title'] = $this->_xmlEntities($url);
6411d47afe1Sandi        $url = str_replace('\\','/',$url);
6421d47afe1Sandi        $url = 'file:///'.$url;
6431d47afe1Sandi        $link['url'] = $url;
6440cecf9d5Sandi
6451d47afe1Sandi        //output formatted
646a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6470cecf9d5Sandi    }
6480cecf9d5Sandi
64971352defSandi    function emaillink($address, $name = NULL) {
65071352defSandi        global $conf;
65171352defSandi        //simple setup
65271352defSandi        $link = array();
65371352defSandi        $link['target'] = '';
65471352defSandi        $link['pre']    = '';
65571352defSandi        $link['suf']   = '';
65671352defSandi        $link['style']  = '';
65771352defSandi        $link['more']   = '';
6580cecf9d5Sandi
659704fb054SAnika Henke        $name = $this->_getLinkTitle($name, $address, $isImage);
6600cecf9d5Sandi        if ( !$isImage ) {
661776b36ecSAndreas Gohr            $link['class']='mail JSnocheck';
6620cecf9d5Sandi        } else {
663776b36ecSAndreas Gohr            $link['class']='media JSnocheck';
6640cecf9d5Sandi        }
6650cecf9d5Sandi
66607738714SAndreas Gohr        $address = $this->_xmlEntities($address);
66700a7b5adSEsther Brunner        $address = obfuscate($address);
66800a7b5adSEsther Brunner        $title   = $address;
6698c128049SAndreas Gohr
67071352defSandi        if(empty($name)){
67100a7b5adSEsther Brunner            $name = $address;
67271352defSandi        }
6738c128049SAndreas Gohr#elseif($isImage{
6748c128049SAndreas Gohr#            $name = $this->_xmlEntities($name);
6758c128049SAndreas Gohr#        }
6760cecf9d5Sandi
677776b36ecSAndreas Gohr        if($conf['mailguard'] == 'visible') $address = rawurlencode($address);
678776b36ecSAndreas Gohr
679776b36ecSAndreas Gohr        $link['url']   = 'mailto:'.$address;
68071352defSandi        $link['name']  = $name;
68171352defSandi        $link['title'] = $title;
6820cecf9d5Sandi
68371352defSandi        //output formatted
684a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6850cecf9d5Sandi    }
6860cecf9d5Sandi
6874826ab45Sandi    function internalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
688dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
68937e34a5eSandi        global $ID;
69091df343aSAndreas Gohr        list($src,$hash) = explode('#',$src,2);
69137e34a5eSandi        resolve_mediaid(getNS($ID),$src, $exists);
6920cecf9d5Sandi
693d98d4540SBen Coburn        $noLink = false;
6948acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
695b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
6963685f775Sandi
697ecebf3a8SAndreas Gohr        list($ext,$mime,$dl) = mimetype($src);
698b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
699dc673a5bSjoe.lapp            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),($linking=='direct'));
7001c882ba8SAndreas Gohr        }elseif($mime == 'application/x-shockwave-flash' && $render){
7012ca14335SEsther Brunner            // don't link flash movies
70244881bd0Shenning.noren            $noLink = true;
70355efc227SAndreas Gohr        }else{
7042ca14335SEsther Brunner            // add file icons
7059d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
7069d2ddea4SAndreas Gohr            $link['class'] .= ' mediafile mf_'.$class;
7076de3759aSAndreas Gohr            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),true);
70855efc227SAndreas Gohr        }
7093685f775Sandi
71091df343aSAndreas Gohr        if($hash) $link['url'] .= '#'.$hash;
71191df343aSAndreas Gohr
7126fe20453SGina Haeussge        //markup non existing files
7136fe20453SGina Haeussge        if (!$exists)
7146fe20453SGina Haeussge          $link['class'] .= ' wikilink2';
7156fe20453SGina Haeussge
7163685f775Sandi        //output formatted
717dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
7182ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
7190cecf9d5Sandi    }
7200cecf9d5Sandi
7214826ab45Sandi    function externalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
722dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
72391df343aSAndreas Gohr        list($src,$hash) = explode('#',$src,2);
724d98d4540SBen Coburn        $noLink = false;
7258acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
726b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
727b739ff0fSPierre Spring
728b739ff0fSPierre Spring        $link['url']    = ml($src,array('cache'=>$cache));
7293685f775Sandi
730ecebf3a8SAndreas Gohr        list($ext,$mime,$dl) = mimetype($src);
731b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
7322ca14335SEsther Brunner             // link only jpeg images
73344881bd0Shenning.noren             // if ($ext != 'jpg' && $ext != 'jpeg') $noLink = true;
7341c882ba8SAndreas Gohr        }elseif($mime == 'application/x-shockwave-flash' && $render){
7352ca14335SEsther Brunner             // don't link flash movies
73644881bd0Shenning.noren             $noLink = true;
7372ca14335SEsther Brunner        }else{
7382ca14335SEsther Brunner             // add file icons
739d15166e5SAndreas Gohr             $link['class'] .= ' mediafile mf_'.$ext;
7402ca14335SEsther Brunner        }
7412ca14335SEsther Brunner
74291df343aSAndreas Gohr        if($hash) $link['url'] .= '#'.$hash;
74391df343aSAndreas Gohr
7443685f775Sandi        //output formatted
745dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
7462ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
7470cecf9d5Sandi    }
7480cecf9d5Sandi
7494826ab45Sandi    /**
7503db95becSAndreas Gohr     * Renders an RSS feed
751b625487dSandi     *
752b625487dSandi     * @author Andreas Gohr <andi@splitbrain.org>
753b625487dSandi     */
7543db95becSAndreas Gohr    function rss ($url,$params){
755b625487dSandi        global $lang;
7563db95becSAndreas Gohr        global $conf;
7573db95becSAndreas Gohr
7583db95becSAndreas Gohr        require_once(DOKU_INC.'inc/FeedParser.php');
7593db95becSAndreas Gohr        $feed = new FeedParser();
76000077af8SAndreas Gohr        $feed->set_feed_url($url);
761b625487dSandi
762b625487dSandi        //disable warning while fetching
763bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { $elvl = error_reporting(E_ERROR); }
7643db95becSAndreas Gohr        $rc = $feed->init();
765bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { error_reporting($elvl); }
766b625487dSandi
7673db95becSAndreas Gohr        //decide on start and end
7683db95becSAndreas Gohr        if($params['reverse']){
7693db95becSAndreas Gohr            $mod = -1;
7703db95becSAndreas Gohr            $start = $feed->get_item_quantity()-1;
7713db95becSAndreas Gohr            $end   = $start - ($params['max']);
772b2a412b0SAndreas Gohr            $end   = ($end < -1) ? -1 : $end;
7733db95becSAndreas Gohr        }else{
7743db95becSAndreas Gohr            $mod   = 1;
7753db95becSAndreas Gohr            $start = 0;
7763db95becSAndreas Gohr            $end   = $feed->get_item_quantity();
7773db95becSAndreas Gohr            $end   = ($end > $params['max']) ? $params['max'] : $end;;
7783db95becSAndreas Gohr        }
7793db95becSAndreas Gohr
780a2d649c4Sandi        $this->doc .= '<ul class="rss">';
7813db95becSAndreas Gohr        if($rc){
7823db95becSAndreas Gohr            for ($x = $start; $x != $end; $x += $mod) {
7831bde1582SAndreas Gohr                $item = $feed->get_item($x);
7843db95becSAndreas Gohr                $this->doc .= '<li><div class="li">';
785d2ea3363SAndreas Gohr                // support feeds without links
786d2ea3363SAndreas Gohr                $lnkurl = $item->get_permalink();
787d2ea3363SAndreas Gohr                if($lnkurl){
7881bde1582SAndreas Gohr                    $this->externallink($item->get_permalink(),
7891bde1582SAndreas Gohr                                        $item->get_title());
790d2ea3363SAndreas Gohr                }else{
791d2ea3363SAndreas Gohr                    $this->doc .= ' '.$item->get_title();
792d2ea3363SAndreas Gohr                }
7933db95becSAndreas Gohr                if($params['author']){
7941bde1582SAndreas Gohr                    $author = $item->get_author(0);
7951bde1582SAndreas Gohr                    if($author){
7961bde1582SAndreas Gohr                        $name = $author->get_name();
7971bde1582SAndreas Gohr                        if(!$name) $name = $author->get_email();
7981bde1582SAndreas Gohr                        if($name) $this->doc .= ' '.$lang['by'].' '.$name;
7991bde1582SAndreas Gohr                    }
8003db95becSAndreas Gohr                }
8013db95becSAndreas Gohr                if($params['date']){
8022e7e0c29SAndreas Gohr                    $this->doc .= ' ('.$item->get_local_date($conf['dformat']).')';
8033db95becSAndreas Gohr                }
8041bde1582SAndreas Gohr                if($params['details']){
8053db95becSAndreas Gohr                    $this->doc .= '<div class="detail">';
806173dccb7STom N Harris                    if($conf['htmlok']){
8071bde1582SAndreas Gohr                        $this->doc .= $item->get_description();
8083db95becSAndreas Gohr                    }else{
8091bde1582SAndreas Gohr                        $this->doc .= strip_tags($item->get_description());
8103db95becSAndreas Gohr                    }
8113db95becSAndreas Gohr                    $this->doc .= '</div>';
8123db95becSAndreas Gohr                }
8133db95becSAndreas Gohr
8143db95becSAndreas Gohr                $this->doc .= '</div></li>';
815b625487dSandi            }
816b625487dSandi        }else{
8173db95becSAndreas Gohr            $this->doc .= '<li><div class="li">';
818a2d649c4Sandi            $this->doc .= '<em>'.$lang['rssfailed'].'</em>';
819b625487dSandi            $this->externallink($url);
82045e147ccSAndreas Gohr            if($conf['allowdebug']){
82145e147ccSAndreas Gohr                $this->doc .= '<!--'.hsc($feed->error).'-->';
82245e147ccSAndreas Gohr            }
8233db95becSAndreas Gohr            $this->doc .= '</div></li>';
824b625487dSandi        }
825a2d649c4Sandi        $this->doc .= '</ul>';
826b625487dSandi    }
827b625487dSandi
8280cecf9d5Sandi    // $numrows not yet implemented
8290cecf9d5Sandi    function table_open($maxcols = NULL, $numrows = NULL){
830b5742cedSPierre Spring        // initialize the row counter used for classes
831b5742cedSPierre Spring        $this->_counter['row_counter'] = 0;
832*91459163SAnika Henke        $this->doc .= '<table class="inline"'.$this->_getPurpleNumberID().'>'.DOKU_LF;
8330cecf9d5Sandi    }
8340cecf9d5Sandi
8350cecf9d5Sandi    function table_close(){
836*91459163SAnika Henke        $this->doc .= '</table>'.$this->_getPurpleNumberLink(1).DOKU_LF;
8370cecf9d5Sandi    }
8380cecf9d5Sandi
8390cecf9d5Sandi    function tablerow_open(){
840b5742cedSPierre Spring        // initialize the cell counter used for classes
841b5742cedSPierre Spring        $this->_counter['cell_counter'] = 0;
842b5742cedSPierre Spring        $class = 'row' . $this->_counter['row_counter']++;
843b5742cedSPierre Spring        $this->doc .= DOKU_TAB . '<tr class="'.$class.'">' . DOKU_LF . DOKU_TAB . DOKU_TAB;
8440cecf9d5Sandi    }
8450cecf9d5Sandi
8460cecf9d5Sandi    function tablerow_close(){
847a2d649c4Sandi        $this->doc .= DOKU_LF . DOKU_TAB . '</tr>' . DOKU_LF;
8480cecf9d5Sandi    }
8490cecf9d5Sandi
8500cecf9d5Sandi    function tableheader_open($colspan = 1, $align = NULL){
851b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
8520cecf9d5Sandi        if ( !is_null($align) ) {
853b5742cedSPierre Spring            $class .= ' '.$align.'align';
8540cecf9d5Sandi        }
855b5742cedSPierre Spring        $class .= '"';
856b5742cedSPierre Spring        $this->doc .= '<th ' . $class;
8570cecf9d5Sandi        if ( $colspan > 1 ) {
858a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
859a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8600cecf9d5Sandi        }
861a2d649c4Sandi        $this->doc .= '>';
8620cecf9d5Sandi    }
8630cecf9d5Sandi
8640cecf9d5Sandi    function tableheader_close(){
865a2d649c4Sandi        $this->doc .= '</th>';
8660cecf9d5Sandi    }
8670cecf9d5Sandi
8680cecf9d5Sandi    function tablecell_open($colspan = 1, $align = NULL){
869b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
8700cecf9d5Sandi        if ( !is_null($align) ) {
871b5742cedSPierre Spring            $class .= ' '.$align.'align';
8720cecf9d5Sandi        }
873b5742cedSPierre Spring        $class .= '"';
874b5742cedSPierre Spring        $this->doc .= '<td '.$class;
8750cecf9d5Sandi        if ( $colspan > 1 ) {
876a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
877a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8780cecf9d5Sandi        }
879a2d649c4Sandi        $this->doc .= '>';
8800cecf9d5Sandi    }
8810cecf9d5Sandi
8820cecf9d5Sandi    function tablecell_close(){
883a2d649c4Sandi        $this->doc .= '</td>';
8840cecf9d5Sandi    }
8850cecf9d5Sandi
8860cecf9d5Sandi    //----------------------------------------------------------
8870cecf9d5Sandi    // Utils
8880cecf9d5Sandi
889ba11bd29Sandi    /**
8903fd0b676Sandi     * Build a link
8913fd0b676Sandi     *
8923fd0b676Sandi     * Assembles all parts defined in $link returns HTML for the link
893ba11bd29Sandi     *
894ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
895ba11bd29Sandi     */
896433bef32Sandi    function _formatLink($link){
897ba11bd29Sandi        //make sure the url is XHTML compliant (skip mailto)
898ba11bd29Sandi        if(substr($link['url'],0,7) != 'mailto:'){
899ba11bd29Sandi            $link['url'] = str_replace('&','&amp;',$link['url']);
900ba11bd29Sandi            $link['url'] = str_replace('&amp;amp;','&amp;',$link['url']);
901ba11bd29Sandi        }
902ba11bd29Sandi        //remove double encodings in titles
903ba11bd29Sandi        $link['title'] = str_replace('&amp;amp;','&amp;',$link['title']);
904ba11bd29Sandi
905453493f2SAndreas Gohr        // be sure there are no bad chars in url or title
906453493f2SAndreas Gohr        // (we can't do this for name because it can contain an img tag)
907453493f2SAndreas Gohr        $link['url']   = strtr($link['url'],array('>'=>'%3E','<'=>'%3C','"'=>'%22'));
908453493f2SAndreas Gohr        $link['title'] = strtr($link['title'],array('>'=>'&gt;','<'=>'&lt;','"'=>'&quot;'));
909453493f2SAndreas Gohr
910ba11bd29Sandi        $ret  = '';
911ba11bd29Sandi        $ret .= $link['pre'];
912ba11bd29Sandi        $ret .= '<a href="'.$link['url'].'"';
913bb4866bdSchris        if(!empty($link['class']))  $ret .= ' class="'.$link['class'].'"';
914bb4866bdSchris        if(!empty($link['target'])) $ret .= ' target="'.$link['target'].'"';
915bb4866bdSchris        if(!empty($link['title']))  $ret .= ' title="'.$link['title'].'"';
916bb4866bdSchris        if(!empty($link['style']))  $ret .= ' style="'.$link['style'].'"';
91744a6b4c7SAndreas Gohr        if(!empty($link['rel']))    $ret .= ' rel="'.$link['rel'].'"';
918bb4866bdSchris        if(!empty($link['more']))   $ret .= ' '.$link['more'];
919ba11bd29Sandi        $ret .= '>';
920ba11bd29Sandi        $ret .= $link['name'];
921ba11bd29Sandi        $ret .= '</a>';
922ba11bd29Sandi        $ret .= $link['suf'];
923ba11bd29Sandi        return $ret;
924ba11bd29Sandi    }
925ba11bd29Sandi
926ba11bd29Sandi    /**
9273fd0b676Sandi     * Renders internal and external media
9283fd0b676Sandi     *
9293fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
9303fd0b676Sandi     */
9313fd0b676Sandi    function _media ($src, $title=NULL, $align=NULL, $width=NULL,
932b739ff0fSPierre Spring                      $height=NULL, $cache=NULL, $render = true) {
9333fd0b676Sandi
9343fd0b676Sandi        $ret = '';
9353fd0b676Sandi
936ecebf3a8SAndreas Gohr        list($ext,$mime,$dl) = mimetype($src);
9373fd0b676Sandi        if(substr($mime,0,5) == 'image'){
938b739ff0fSPierre Spring            // first get the $title
939b739ff0fSPierre Spring            if (!is_null($title)) {
940b739ff0fSPierre Spring                $title  = $this->_xmlEntities($title);
941b739ff0fSPierre Spring            }elseif($ext == 'jpg' || $ext == 'jpeg'){
942b739ff0fSPierre Spring                //try to use the caption from IPTC/EXIF
943b739ff0fSPierre Spring                require_once(DOKU_INC.'inc/JpegMeta.php');
944b739ff0fSPierre Spring                $jpeg =& new JpegMeta(mediaFN($src));
945b739ff0fSPierre Spring                if($jpeg !== false) $cap = $jpeg->getTitle();
946b739ff0fSPierre Spring                if($cap){
947b739ff0fSPierre Spring                    $title = $this->_xmlEntities($cap);
948b739ff0fSPierre Spring                }
949b739ff0fSPierre Spring            }
950b739ff0fSPierre Spring            if (!$render) {
951b739ff0fSPierre Spring                // if the picture is not supposed to be rendered
952b739ff0fSPierre Spring                // return the title of the picture
953b739ff0fSPierre Spring                if (!$title) {
954b739ff0fSPierre Spring                    // just show the sourcename
955b739ff0fSPierre Spring                    $title = $this->_xmlEntities(basename(noNS($src)));
956b739ff0fSPierre Spring                }
957b739ff0fSPierre Spring                return $title;
958b739ff0fSPierre Spring            }
9593fd0b676Sandi            //add image tag
9606de3759aSAndreas Gohr            $ret .= '<img src="'.ml($src,array('w'=>$width,'h'=>$height,'cache'=>$cache)).'"';
9613fd0b676Sandi            $ret .= ' class="media'.$align.'"';
9623fd0b676Sandi
9634ab889eaSAndreas Gohr            // make left/right alignment for no-CSS view work (feeds)
9644ab889eaSAndreas Gohr            if($align == 'right') $ret .= ' align="right"';
9654ab889eaSAndreas Gohr            if($align == 'left')  $ret .= ' align="left"';
9664ab889eaSAndreas Gohr
967b739ff0fSPierre Spring            if ($title) {
968b739ff0fSPierre Spring                $ret .= ' title="' . $title . '"';
969b739ff0fSPierre Spring                $ret .= ' alt="'   . $title .'"';
9703fd0b676Sandi            }else{
9713fd0b676Sandi                $ret .= ' alt=""';
9723fd0b676Sandi            }
9733fd0b676Sandi
9743fd0b676Sandi            if ( !is_null($width) )
9753fd0b676Sandi                $ret .= ' width="'.$this->_xmlEntities($width).'"';
9763fd0b676Sandi
9773fd0b676Sandi            if ( !is_null($height) )
9783fd0b676Sandi                $ret .= ' height="'.$this->_xmlEntities($height).'"';
9793fd0b676Sandi
9803fd0b676Sandi            $ret .= ' />';
9813fd0b676Sandi
9823fd0b676Sandi        }elseif($mime == 'application/x-shockwave-flash'){
9831c882ba8SAndreas Gohr            if (!$render) {
9841c882ba8SAndreas Gohr                // if the flash is not supposed to be rendered
9851c882ba8SAndreas Gohr                // return the title of the flash
9861c882ba8SAndreas Gohr                if (!$title) {
9871c882ba8SAndreas Gohr                    // just show the sourcename
98807bf32b2SAndreas Gohr                    $title = basename(noNS($src));
9891c882ba8SAndreas Gohr                }
99007bf32b2SAndreas Gohr                return $this->_xmlEntities($title);
9911c882ba8SAndreas Gohr            }
9921c882ba8SAndreas Gohr
99307bf32b2SAndreas Gohr            $att = array();
99407bf32b2SAndreas Gohr            $att['class'] = "media$align";
99507bf32b2SAndreas Gohr            if($align == 'right') $att['align'] = 'right';
99607bf32b2SAndreas Gohr            if($align == 'left')  $att['align'] = 'left';
997109577f5SAndreas Gohr            $ret .= html_flashobject(ml($src,array('cache'=>$cache)),$width,$height,
99807bf32b2SAndreas Gohr                                     array('quality' => 'high'),
99907bf32b2SAndreas Gohr                                     null,
100007bf32b2SAndreas Gohr                                     $att,
100107bf32b2SAndreas Gohr                                     $this->_xmlEntities($title));
10020f428d7dSAndreas Gohr        }elseif($title){
10033fd0b676Sandi            // well at least we have a title to display
10043fd0b676Sandi            $ret .= $this->_xmlEntities($title);
10053fd0b676Sandi        }else{
10065291ca3aSAndreas Gohr            // just show the sourcename
10070f428d7dSAndreas Gohr            $ret .= $this->_xmlEntities(basename(noNS($src)));
10083fd0b676Sandi        }
10093fd0b676Sandi
10103fd0b676Sandi        return $ret;
10113fd0b676Sandi    }
10123fd0b676Sandi
1013433bef32Sandi    function _xmlEntities($string) {
1014de117061Schris        return htmlspecialchars($string,ENT_QUOTES,'UTF-8');
10150cecf9d5Sandi    }
10160cecf9d5Sandi
10178a831f2bSAndreas Gohr    /**
10188a831f2bSAndreas Gohr     * Creates a linkid from a headline
1019c5a8fd96SAndreas Gohr     *
1020c5a8fd96SAndreas Gohr     * @param string  $title   The headline title
1021c5a8fd96SAndreas Gohr     * @param boolean $create  Create a new unique ID?
1022c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
10238a831f2bSAndreas Gohr     */
1024c5a8fd96SAndreas Gohr    function _headerToLink($title,$create=false) {
1025c5a8fd96SAndreas Gohr        if($create){
10264ceab83fSAndreas Gohr            return sectionID($title,$this->headers);
10274ceab83fSAndreas Gohr        }else{
1028443d207bSAndreas Gohr            $check = false;
1029443d207bSAndreas Gohr            return sectionID($title,$check);
1030c5a8fd96SAndreas Gohr        }
10310cecf9d5Sandi    }
10320cecf9d5Sandi
1033af587fa8Sandi    /**
10343fd0b676Sandi     * Construct a title and handle images in titles
10353fd0b676Sandi     *
10360b7c14c2Sandi     * @author Harry Fuecks <hfuecks@gmail.com>
10373fd0b676Sandi     */
1038fe9ec250SChris Smith    function _getLinkTitle($title, $default, & $isImage, $id=NULL, $linktype='content') {
1039bb0a59d4Sjan        global $conf;
1040bb0a59d4Sjan
104144881bd0Shenning.noren        $isImage = false;
1042a6f3bd20SAnika Henke        if ( is_null($title) || trim($title)=='') {
1043fe9ec250SChris Smith            if (useHeading($linktype) && $id) {
1044fc18c0fbSchris                $heading = p_get_first_heading($id,true);
1045bb0a59d4Sjan                if ($heading) {
1046433bef32Sandi                    return $this->_xmlEntities($heading);
1047bb0a59d4Sjan                }
1048bb0a59d4Sjan            }
1049433bef32Sandi            return $this->_xmlEntities($default);
10500cecf9d5Sandi        } else if ( is_array($title) ) {
105144881bd0Shenning.noren            $isImage = true;
1052433bef32Sandi            return $this->_imageTitle($title);
105368c26e6dSMichael Klier        } else {
105468c26e6dSMichael Klier            return $this->_xmlEntities($title);
10550cecf9d5Sandi        }
10560cecf9d5Sandi    }
10570cecf9d5Sandi
10580cecf9d5Sandi    /**
10593fd0b676Sandi     * Returns an HTML code for images used in link titles
10603fd0b676Sandi     *
10613fd0b676Sandi     * @todo Resolve namespace on internal images
10623fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
10630cecf9d5Sandi     */
1064433bef32Sandi    function _imageTitle($img) {
1065433bef32Sandi        return $this->_media($img['src'],
10664826ab45Sandi                              $img['title'],
10674826ab45Sandi                              $img['align'],
10684826ab45Sandi                              $img['width'],
10694826ab45Sandi                              $img['height'],
10704826ab45Sandi                              $img['cache']);
10710cecf9d5Sandi    }
1072b739ff0fSPierre Spring
1073b739ff0fSPierre Spring    /**
1074b739ff0fSPierre Spring     * _getMediaLinkConf is a helperfunction to internalmedia() and externalmedia()
1075b739ff0fSPierre Spring     * which returns a basic link to a media.
1076b739ff0fSPierre Spring     *
1077b739ff0fSPierre Spring     * @author Pierre Spring <pierre.spring@liip.ch>
1078b739ff0fSPierre Spring     * @param string $src
1079b739ff0fSPierre Spring     * @param string $title
1080b739ff0fSPierre Spring     * @param string $align
1081b739ff0fSPierre Spring     * @param string $width
1082b739ff0fSPierre Spring     * @param string $height
1083b739ff0fSPierre Spring     * @param string $cache
1084b739ff0fSPierre Spring     * @param string $render
1085b739ff0fSPierre Spring     * @access protected
1086b739ff0fSPierre Spring     * @return array
1087b739ff0fSPierre Spring     */
1088b739ff0fSPierre Spring    function _getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render)
1089b739ff0fSPierre Spring    {
1090b739ff0fSPierre Spring        global $conf;
1091b739ff0fSPierre Spring
1092b739ff0fSPierre Spring        $link = array();
1093b739ff0fSPierre Spring        $link['class']  = 'media';
1094b739ff0fSPierre Spring        $link['style']  = '';
1095b739ff0fSPierre Spring        $link['pre']    = '';
1096b739ff0fSPierre Spring        $link['suf']    = '';
1097b739ff0fSPierre Spring        $link['more']   = '';
1098b739ff0fSPierre Spring        $link['target'] = $conf['target']['media'];
1099b739ff0fSPierre Spring        $link['title']  = $this->_xmlEntities($src);
1100b739ff0fSPierre Spring        $link['name']   = $this->_media($src, $title, $align, $width, $height, $cache, $render);
1101b739ff0fSPierre Spring
1102b739ff0fSPierre Spring        return $link;
1103b739ff0fSPierre Spring    }
1104*91459163SAnika Henke
1105*91459163SAnika Henke    /**
1106*91459163SAnika Henke     * Builds unique Hierarchical ID:
1107*91459163SAnika Henke     * If $conf['purplenumbers'] is 2, it is unique per site,
1108*91459163SAnika Henke     * otherwise it is unique per page.
1109*91459163SAnika Henke     *
1110*91459163SAnika Henke     * @author Anika Henke <anika@selfthinker.org>
1111*91459163SAnika Henke     */
1112*91459163SAnika Henke    function _getHID($noprefix=0) {
1113*91459163SAnika Henke        global $conf;
1114*91459163SAnika Henke        if ($noprefix) {
1115*91459163SAnika Henke            $prefix = '';
1116*91459163SAnika Henke        } else if ($conf['purplenumbers']==2) {
1117*91459163SAnika Henke            global $ID;
1118*91459163SAnika Henke            $prefix = $ID.'.';
1119*91459163SAnika Henke        } else {
1120*91459163SAnika Henke            $prefix = 'HID';
1121*91459163SAnika Henke        }
1122*91459163SAnika Henke        return $prefix.rtrim(join('.',$this->node),'.0').rtrim(':'.$this->pnid,':0');
1123*91459163SAnika Henke    }
1124*91459163SAnika Henke
1125*91459163SAnika Henke    /**
1126*91459163SAnika Henke     * Equips each designated element with a Purple Number (Hierarchical ID).
1127*91459163SAnika Henke     *
1128*91459163SAnika Henke     * @author Anika Henke <anika@selfthinker.org>
1129*91459163SAnika Henke     */
1130*91459163SAnika Henke    function _getPurpleNumberID() {
1131*91459163SAnika Henke        global $conf;
1132*91459163SAnika Henke        $this->pnid++;
1133*91459163SAnika Henke        if ($conf['purplenumbers']) {
1134*91459163SAnika Henke            return ' id="'.$this->_getHID().'"';
1135*91459163SAnika Henke        }
1136*91459163SAnika Henke        return '';
1137*91459163SAnika Henke    }
1138*91459163SAnika Henke
1139*91459163SAnika Henke    /**
1140*91459163SAnika Henke     * Creates a link to the current Purple Number (Hierarchical ID).
1141*91459163SAnika Henke     * If the link cannot be inside its corresponding element (e.g. tables),
1142*91459163SAnika Henke     * $outside is set and p.pnlink is added around the link.
1143*91459163SAnika Henke     *
1144*91459163SAnika Henke     * @author Anika Henke <anika@selfthinker.org>
1145*91459163SAnika Henke     */
1146*91459163SAnika Henke    function _getPurpleNumberLink($outside=0) {
1147*91459163SAnika Henke        global $conf;
1148*91459163SAnika Henke        if ($conf['purplenumbers']) {
1149*91459163SAnika Henke            global $lang;
1150*91459163SAnika Henke            $pnlink = '<a href="#'.$this->_getHID().'" class="pn" title="'.$lang['sectionlink'].'">¶</a>';
1151*91459163SAnika Henke            if ($outside) {
1152*91459163SAnika Henke                return '<p class="pnlink">'.$pnlink.'</p>';
1153*91459163SAnika Henke            }
1154*91459163SAnika Henke            return ' <!--PN-->'.$pnlink;
1155*91459163SAnika Henke        }
1156*91459163SAnika Henke        return '';
1157*91459163SAnika Henke    }
1158*91459163SAnika Henke
11590cecf9d5Sandi}
11600cecf9d5Sandi
11614826ab45Sandi//Setup VIM: ex: et ts=4 enc=utf-8 :
1162