xref: /dokuwiki/inc/parser/xhtml.php (revision fe9ec250a6558c8352a35b6537cdc30d6c9f5477)
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
900976812SAndreas Gohrif(!defined('DOKU_INC')) define('DOKU_INC',fullpath(dirname(__FILE__).'/../../').'/');
100cecf9d5Sandi
110cecf9d5Sandiif ( !defined('DOKU_LF') ) {
120cecf9d5Sandi    // Some whitespace to help View > Source
130cecf9d5Sandi    define ('DOKU_LF',"\n");
140cecf9d5Sandi}
150cecf9d5Sandi
160cecf9d5Sandiif ( !defined('DOKU_TAB') ) {
170cecf9d5Sandi    // Some whitespace to help View > Source
180cecf9d5Sandi    define ('DOKU_TAB',"\t");
190cecf9d5Sandi}
200cecf9d5Sandi
210e1c636eSandirequire_once DOKU_INC . 'inc/parser/renderer.php';
2256fa9a3fSAndreas Gohrrequire_once DOKU_INC . 'inc/html.php';
230e1c636eSandi
240cecf9d5Sandi/**
25b625487dSandi * The Renderer
260cecf9d5Sandi */
27ac83b9d8Sandiclass Doku_Renderer_xhtml extends Doku_Renderer {
280cecf9d5Sandi
29c5a8fd96SAndreas Gohr    // @access public
30c5a8fd96SAndreas Gohr    var $doc = '';        // will contain the whole document
31c5a8fd96SAndreas Gohr    var $toc = array();   // will contain the Table of Contents
32c5a8fd96SAndreas Gohr
330cecf9d5Sandi
340cecf9d5Sandi    var $headers = array();
350cecf9d5Sandi    var $footnotes = array();
36af587fa8Sandi    var $lastsec = 0;
377764a90aSandi    var $store = '';
387764a90aSandi
39b5742cedSPierre Spring    var $_counter = array(); // used as global counter, introduced for table classes
40b5742cedSPierre Spring
415f70445dSAndreas Gohr    function getFormat(){
425f70445dSAndreas Gohr        return 'xhtml';
435f70445dSAndreas Gohr    }
445f70445dSAndreas Gohr
455f70445dSAndreas Gohr
460cecf9d5Sandi    function document_start() {
47c5a8fd96SAndreas Gohr        //reset some internals
48c5a8fd96SAndreas Gohr        $this->toc     = array();
49c5a8fd96SAndreas Gohr        $this->headers = array();
500cecf9d5Sandi    }
510cecf9d5Sandi
520cecf9d5Sandi    function document_end() {
530cecf9d5Sandi        if ( count ($this->footnotes) > 0 ) {
54a2d649c4Sandi            $this->doc .= '<div class="footnotes">'.DOKU_LF;
55d74aace9Schris
56d74aace9Schris            $id = 0;
570cecf9d5Sandi            foreach ( $this->footnotes as $footnote ) {
58d74aace9Schris                $id++;   // the number of the current footnote
59d74aace9Schris
60d74aace9Schris                // check its not a placeholder that indicates actual footnote text is elsewhere
61d74aace9Schris                if (substr($footnote, 0, 5) != "@@FNT") {
62d74aace9Schris
63d74aace9Schris                    // open the footnote and set the anchor and backlink
64d74aace9Schris                    $this->doc .= '<div class="fn">';
6529bfcd16SAndreas Gohr                    $this->doc .= '<sup><a href="#fnt__'.$id.'" id="fn__'.$id.'" name="fn__'.$id.'" class="fn_bot">';
6629bfcd16SAndreas Gohr                    $this->doc .= $id.')</a></sup> '.DOKU_LF;
67d74aace9Schris
68d74aace9Schris                    // get any other footnotes that use the same markup
69d74aace9Schris                    $alt = array_keys($this->footnotes, "@@FNT$id");
70d74aace9Schris
71d74aace9Schris                    if (count($alt)) {
72d74aace9Schris                      foreach ($alt as $ref) {
73d74aace9Schris                        // set anchor and backlink for the other footnotes
7429bfcd16SAndreas Gohr                        $this->doc .= ', <sup><a href="#fnt__'.($ref+1).'" id="fn__'.($ref+1).'" name="fn__'.($ref+1).'" class="fn_bot">';
7529bfcd16SAndreas Gohr                        $this->doc .= ($ref+1).')</a></sup> '.DOKU_LF;
76d74aace9Schris                      }
77d74aace9Schris                    }
78d74aace9Schris
79d74aace9Schris                    // add footnote markup and close this footnote
80a2d649c4Sandi                    $this->doc .= $footnote;
81d74aace9Schris                    $this->doc .= '</div>' . DOKU_LF;
82d74aace9Schris                }
830cecf9d5Sandi            }
84a2d649c4Sandi            $this->doc .= '</div>'.DOKU_LF;
850cecf9d5Sandi        }
86c5a8fd96SAndreas Gohr
87b8595a66SAndreas Gohr        // Prepare the TOC
88b8595a66SAndreas Gohr        if($this->info['toc'] && is_array($this->toc) && count($this->toc) > 2){
89b8595a66SAndreas Gohr            global $TOC;
90b8595a66SAndreas Gohr            $TOC = $this->toc;
910cecf9d5Sandi        }
923e55d035SAndreas Gohr
933e55d035SAndreas Gohr        // make sure there are no empty paragraphs
9427918226Schris        $this->doc = preg_replace('#<p>\s*</p>#','',$this->doc);
95e41c4da9SAndreas Gohr    }
960cecf9d5Sandi
97e7856beaSchris    function toc_additem($id, $text, $level) {
98af587fa8Sandi        global $conf;
99af587fa8Sandi
100c5a8fd96SAndreas Gohr        //handle TOC
101c5a8fd96SAndreas Gohr        if($level >= $conf['toptoclevel'] && $level <= $conf['maxtoclevel']){
1027d91652aSAndreas Gohr            $this->toc[] = html_mktocitem($id, $text, $level-$conf['toptoclevel']+1);
103c5a8fd96SAndreas Gohr        }
104e7856beaSchris    }
105e7856beaSchris
106e7856beaSchris    function header($text, $level, $pos) {
107e7856beaSchris
108e7856beaSchris        $hid = $this->_headerToLink($text,true);
109e7856beaSchris
110e7856beaSchris        //only add items within configured levels
111e7856beaSchris        $this->toc_additem($hid, $text, $level);
112c5a8fd96SAndreas Gohr
113c5a8fd96SAndreas Gohr        // write the header
114c5a8fd96SAndreas Gohr        $this->doc .= DOKU_LF.'<h'.$level.'><a name="'.$hid.'" id="'.$hid.'">';
115a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
116c6a1c1bfSAnika Henke        $this->doc .= "</a></h$level>".DOKU_LF;
1170cecf9d5Sandi    }
1180cecf9d5Sandi
11935dae8b0SBen Coburn     /**
12035dae8b0SBen Coburn     * Section edit marker is replaced by an edit button when
12135dae8b0SBen Coburn     * the page is editable. Replacement done in 'inc/html.php#html_secedit'
12235dae8b0SBen Coburn     *
12335dae8b0SBen Coburn     * @author Andreas Gohr <andi@splitbrain.org>
12435dae8b0SBen Coburn     * @author Ben Coburn   <btcoburn@silicodon.net>
12535dae8b0SBen Coburn     */
12635dae8b0SBen Coburn    function section_edit($start, $end, $level, $name) {
12735dae8b0SBen Coburn        global $conf;
12835dae8b0SBen Coburn
12935dae8b0SBen Coburn        if ($start!=-1 && $level<=$conf['maxseclevel']) {
13035dae8b0SBen Coburn            $name = str_replace('"', '', $name);
13135dae8b0SBen Coburn            $this->doc .= '<!-- SECTION "'.$name.'" ['.$start.'-'.(($end===0)?'':$end).'] -->';
13235dae8b0SBen Coburn        }
13335dae8b0SBen Coburn    }
13435dae8b0SBen Coburn
1350cecf9d5Sandi    function section_open($level) {
136a2d649c4Sandi        $this->doc .= "<div class=\"level$level\">".DOKU_LF;
1370cecf9d5Sandi    }
1380cecf9d5Sandi
1390cecf9d5Sandi    function section_close() {
140a2d649c4Sandi        $this->doc .= DOKU_LF.'</div>'.DOKU_LF;
1410cecf9d5Sandi    }
1420cecf9d5Sandi
1430cecf9d5Sandi    function cdata($text) {
144a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
1450cecf9d5Sandi    }
1460cecf9d5Sandi
1470cecf9d5Sandi    function p_open() {
148a2d649c4Sandi        $this->doc .= DOKU_LF.'<p>'.DOKU_LF;
1490cecf9d5Sandi    }
1500cecf9d5Sandi
1510cecf9d5Sandi    function p_close() {
152a2d649c4Sandi        $this->doc .= DOKU_LF.'</p>'.DOKU_LF;
1530cecf9d5Sandi    }
1540cecf9d5Sandi
1550cecf9d5Sandi    function linebreak() {
156a2d649c4Sandi        $this->doc .= '<br/>'.DOKU_LF;
1570cecf9d5Sandi    }
1580cecf9d5Sandi
1590cecf9d5Sandi    function hr() {
1604beabca9SAnika Henke        $this->doc .= '<hr />'.DOKU_LF;
1610cecf9d5Sandi    }
1620cecf9d5Sandi
1630cecf9d5Sandi    function strong_open() {
164a2d649c4Sandi        $this->doc .= '<strong>';
1650cecf9d5Sandi    }
1660cecf9d5Sandi
1670cecf9d5Sandi    function strong_close() {
168a2d649c4Sandi        $this->doc .= '</strong>';
1690cecf9d5Sandi    }
1700cecf9d5Sandi
1710cecf9d5Sandi    function emphasis_open() {
172a2d649c4Sandi        $this->doc .= '<em>';
1730cecf9d5Sandi    }
1740cecf9d5Sandi
1750cecf9d5Sandi    function emphasis_close() {
176a2d649c4Sandi        $this->doc .= '</em>';
1770cecf9d5Sandi    }
1780cecf9d5Sandi
1790cecf9d5Sandi    function underline_open() {
18002e51121SAnika Henke        $this->doc .= '<em class="u">';
1810cecf9d5Sandi    }
1820cecf9d5Sandi
1830cecf9d5Sandi    function underline_close() {
18402e51121SAnika Henke        $this->doc .= '</em>';
1850cecf9d5Sandi    }
1860cecf9d5Sandi
1870cecf9d5Sandi    function monospace_open() {
188a2d649c4Sandi        $this->doc .= '<code>';
1890cecf9d5Sandi    }
1900cecf9d5Sandi
1910cecf9d5Sandi    function monospace_close() {
192a2d649c4Sandi        $this->doc .= '</code>';
1930cecf9d5Sandi    }
1940cecf9d5Sandi
1950cecf9d5Sandi    function subscript_open() {
196a2d649c4Sandi        $this->doc .= '<sub>';
1970cecf9d5Sandi    }
1980cecf9d5Sandi
1990cecf9d5Sandi    function subscript_close() {
200a2d649c4Sandi        $this->doc .= '</sub>';
2010cecf9d5Sandi    }
2020cecf9d5Sandi
2030cecf9d5Sandi    function superscript_open() {
204a2d649c4Sandi        $this->doc .= '<sup>';
2050cecf9d5Sandi    }
2060cecf9d5Sandi
2070cecf9d5Sandi    function superscript_close() {
208a2d649c4Sandi        $this->doc .= '</sup>';
2090cecf9d5Sandi    }
2100cecf9d5Sandi
2110cecf9d5Sandi    function deleted_open() {
212a2d649c4Sandi        $this->doc .= '<del>';
2130cecf9d5Sandi    }
2140cecf9d5Sandi
2150cecf9d5Sandi    function deleted_close() {
216a2d649c4Sandi        $this->doc .= '</del>';
2170cecf9d5Sandi    }
2180cecf9d5Sandi
2193fd0b676Sandi    /**
2203fd0b676Sandi     * Callback for footnote start syntax
2213fd0b676Sandi     *
2223fd0b676Sandi     * All following content will go to the footnote instead of
223d74aace9Schris     * the document. To achieve this the previous rendered content
2243fd0b676Sandi     * is moved to $store and $doc is cleared
2253fd0b676Sandi     *
2263fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
2273fd0b676Sandi     */
2280cecf9d5Sandi    function footnote_open() {
2297764a90aSandi
2307764a90aSandi        // move current content to store and record footnote
2317764a90aSandi        $this->store = $this->doc;
2327764a90aSandi        $this->doc   = '';
2330cecf9d5Sandi    }
2340cecf9d5Sandi
2353fd0b676Sandi    /**
2363fd0b676Sandi     * Callback for footnote end syntax
2373fd0b676Sandi     *
2383fd0b676Sandi     * All rendered content is moved to the $footnotes array and the old
2393fd0b676Sandi     * content is restored from $store again
2403fd0b676Sandi     *
2413fd0b676Sandi     * @author Andreas Gohr
2423fd0b676Sandi     */
2430cecf9d5Sandi    function footnote_close() {
2447764a90aSandi
245d74aace9Schris        // recover footnote into the stack and restore old content
246d74aace9Schris        $footnote = $this->doc;
2477764a90aSandi        $this->doc = $this->store;
2487764a90aSandi        $this->store = '';
249d74aace9Schris
250d74aace9Schris        // check to see if this footnote has been seen before
251d74aace9Schris        $i = array_search($footnote, $this->footnotes);
252d74aace9Schris
253d74aace9Schris        if ($i === false) {
254d74aace9Schris            // its a new footnote, add it to the $footnotes array
255d74aace9Schris            $id = count($this->footnotes)+1;
256d74aace9Schris            $this->footnotes[count($this->footnotes)] = $footnote;
257d74aace9Schris        } else {
258d74aace9Schris            // seen this one before, translate the index to an id and save a placeholder
259d74aace9Schris            $i++;
260d74aace9Schris            $id = count($this->footnotes)+1;
261d74aace9Schris            $this->footnotes[count($this->footnotes)] = "@@FNT".($i);
262d74aace9Schris        }
263d74aace9Schris
2646b379cbfSAndreas Gohr        // output the footnote reference and link
26529bfcd16SAndreas Gohr        $this->doc .= '<sup><a href="#fn__'.$id.'" name="fnt__'.$id.'" id="fnt__'.$id.'" class="fn_top">'.$id.')</a></sup>';
2660cecf9d5Sandi    }
2670cecf9d5Sandi
2680cecf9d5Sandi    function listu_open() {
269a2d649c4Sandi        $this->doc .= '<ul>'.DOKU_LF;
2700cecf9d5Sandi    }
2710cecf9d5Sandi
2720cecf9d5Sandi    function listu_close() {
273a2d649c4Sandi        $this->doc .= '</ul>'.DOKU_LF;
2740cecf9d5Sandi    }
2750cecf9d5Sandi
2760cecf9d5Sandi    function listo_open() {
277a2d649c4Sandi        $this->doc .= '<ol>'.DOKU_LF;
2780cecf9d5Sandi    }
2790cecf9d5Sandi
2800cecf9d5Sandi    function listo_close() {
281a2d649c4Sandi        $this->doc .= '</ol>'.DOKU_LF;
2820cecf9d5Sandi    }
2830cecf9d5Sandi
2840cecf9d5Sandi    function listitem_open($level) {
285a2d649c4Sandi        $this->doc .= '<li class="level'.$level.'">';
2860cecf9d5Sandi    }
2870cecf9d5Sandi
2880cecf9d5Sandi    function listitem_close() {
289a2d649c4Sandi        $this->doc .= '</li>'.DOKU_LF;
2900cecf9d5Sandi    }
2910cecf9d5Sandi
2920cecf9d5Sandi    function listcontent_open() {
29390db23d7Schris        $this->doc .= '<div class="li">';
2940cecf9d5Sandi    }
2950cecf9d5Sandi
2960cecf9d5Sandi    function listcontent_close() {
29790db23d7Schris        $this->doc .= '</div>'.DOKU_LF;
2980cecf9d5Sandi    }
2990cecf9d5Sandi
3000cecf9d5Sandi    function unformatted($text) {
301a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
3020cecf9d5Sandi    }
3030cecf9d5Sandi
3040cecf9d5Sandi    /**
3053fd0b676Sandi     * Execute PHP code if allowed
3063fd0b676Sandi     *
3075d568b99SChris Smith     * @param  string   $wrapper   html element to wrap result if $conf['phpok'] is okff
3085d568b99SChris Smith     *
3093fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3100cecf9d5Sandi     */
3115d568b99SChris Smith    function php($text, $wrapper='code') {
31235a56260SChris Smith        global $conf;
31335a56260SChris Smith
314d86d5af0SChris Smith        if($conf['phpok']){
315bad0b545Sandi          ob_start();
3164de671bcSandi          eval($text);
3173fd0b676Sandi          $this->doc .= ob_get_contents();
318bad0b545Sandi          ob_end_clean();
319d86d5af0SChris Smith        } else {
3205d568b99SChris Smith          $this->doc .= p_xhtml_cached_geshi($text, 'php', $wrapper);
321d86d5af0SChris Smith        }
3220cecf9d5Sandi    }
3230cecf9d5Sandi
32407f89c3cSAnika Henke    function phpblock($text) {
3255d568b99SChris Smith        $this->php($text, 'pre');
32607f89c3cSAnika Henke    }
32707f89c3cSAnika Henke
3280cecf9d5Sandi    /**
3293fd0b676Sandi     * Insert HTML if allowed
3303fd0b676Sandi     *
3315d568b99SChris Smith     * @param  string   $wrapper   html element to wrap result if $conf['htmlok'] is okff
3325d568b99SChris Smith     *
3333fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3340cecf9d5Sandi     */
3355d568b99SChris Smith    function html($text, $wrapper='code') {
33635a56260SChris Smith        global $conf;
33735a56260SChris Smith
338d86d5af0SChris Smith        if($conf['htmlok']){
339a2d649c4Sandi          $this->doc .= $text;
340d86d5af0SChris Smith        } else {
3415d568b99SChris Smith          $this->doc .= p_xhtml_cached_geshi($text, 'html4strict', $wrapper);
342d86d5af0SChris Smith        }
3434de671bcSandi    }
3440cecf9d5Sandi
34507f89c3cSAnika Henke    function htmlblock($text) {
3465d568b99SChris Smith        $this->html($text, 'pre');
34707f89c3cSAnika Henke    }
34807f89c3cSAnika Henke
3490cecf9d5Sandi    function preformatted($text) {
350a2d649c4Sandi        $this->doc .= '<pre class="code">' . $this->_xmlEntities($text) . '</pre>'. DOKU_LF;
3510cecf9d5Sandi    }
3520cecf9d5Sandi
3530cecf9d5Sandi    function file($text) {
354a2d649c4Sandi        $this->doc .= '<pre class="file">' . $this->_xmlEntities($text). '</pre>'. DOKU_LF;
3550cecf9d5Sandi    }
3560cecf9d5Sandi
3570cecf9d5Sandi    function quote_open() {
35896331712SAnika Henke        $this->doc .= '<blockquote><div class="no">'.DOKU_LF;
3590cecf9d5Sandi    }
3600cecf9d5Sandi
3610cecf9d5Sandi    function quote_close() {
36296331712SAnika Henke        $this->doc .= '</div></blockquote>'.DOKU_LF;
3630cecf9d5Sandi    }
3640cecf9d5Sandi
3650cecf9d5Sandi    /**
3663fd0b676Sandi     * Callback for code text
3673fd0b676Sandi     *
3683fd0b676Sandi     * Uses GeSHi to highlight language syntax
3693fd0b676Sandi     *
3703fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3710cecf9d5Sandi     */
3720cecf9d5Sandi    function code($text, $language = NULL) {
3734de671bcSandi        global $conf;
3740cecf9d5Sandi
3750cecf9d5Sandi        if ( is_null($language) ) {
3760cecf9d5Sandi            $this->preformatted($text);
3770cecf9d5Sandi        } else {
3788f7d700cSchris            $this->doc .= p_xhtml_cached_geshi($text, $language);
3790cecf9d5Sandi        }
3800cecf9d5Sandi    }
3810cecf9d5Sandi
3820cecf9d5Sandi    function acronym($acronym) {
3830cecf9d5Sandi
3840cecf9d5Sandi        if ( array_key_exists($acronym, $this->acronyms) ) {
3850cecf9d5Sandi
386433bef32Sandi            $title = $this->_xmlEntities($this->acronyms[$acronym]);
3870cecf9d5Sandi
388a2d649c4Sandi            $this->doc .= '<acronym title="'.$title
389433bef32Sandi                .'">'.$this->_xmlEntities($acronym).'</acronym>';
3900cecf9d5Sandi
3910cecf9d5Sandi        } else {
392a2d649c4Sandi            $this->doc .= $this->_xmlEntities($acronym);
3930cecf9d5Sandi        }
3940cecf9d5Sandi    }
3950cecf9d5Sandi
3960cecf9d5Sandi    function smiley($smiley) {
3970cecf9d5Sandi        if ( array_key_exists($smiley, $this->smileys) ) {
398433bef32Sandi            $title = $this->_xmlEntities($this->smileys[$smiley]);
399f62ea8a1Sandi            $this->doc .= '<img src="'.DOKU_BASE.'lib/images/smileys/'.$this->smileys[$smiley].
4004beabca9SAnika Henke                '" class="middle" alt="'.
401433bef32Sandi                    $this->_xmlEntities($smiley).'" />';
4020cecf9d5Sandi        } else {
403a2d649c4Sandi            $this->doc .= $this->_xmlEntities($smiley);
4040cecf9d5Sandi        }
4050cecf9d5Sandi    }
4060cecf9d5Sandi
407f62ea8a1Sandi    /*
4084de671bcSandi    * not used
4090cecf9d5Sandi    function wordblock($word) {
4100cecf9d5Sandi        if ( array_key_exists($word, $this->badwords) ) {
411a2d649c4Sandi            $this->doc .= '** BLEEP **';
4120cecf9d5Sandi        } else {
413a2d649c4Sandi            $this->doc .= $this->_xmlEntities($word);
4140cecf9d5Sandi        }
4150cecf9d5Sandi    }
4164de671bcSandi    */
4170cecf9d5Sandi
4180cecf9d5Sandi    function entity($entity) {
4190cecf9d5Sandi        if ( array_key_exists($entity, $this->entities) ) {
420a2d649c4Sandi            $this->doc .= $this->entities[$entity];
4210cecf9d5Sandi        } else {
422a2d649c4Sandi            $this->doc .= $this->_xmlEntities($entity);
4230cecf9d5Sandi        }
4240cecf9d5Sandi    }
4250cecf9d5Sandi
4260cecf9d5Sandi    function multiplyentity($x, $y) {
427a2d649c4Sandi        $this->doc .= "$x&times;$y";
4280cecf9d5Sandi    }
4290cecf9d5Sandi
4300cecf9d5Sandi    function singlequoteopening() {
43171b40da2SAnika Henke        global $lang;
43271b40da2SAnika Henke        $this->doc .= $lang['singlequoteopening'];
4330cecf9d5Sandi    }
4340cecf9d5Sandi
4350cecf9d5Sandi    function singlequoteclosing() {
43671b40da2SAnika Henke        global $lang;
43771b40da2SAnika Henke        $this->doc .= $lang['singlequoteclosing'];
4380cecf9d5Sandi    }
4390cecf9d5Sandi
44057d757d1SAndreas Gohr    function apostrophe() {
44157d757d1SAndreas Gohr        global $lang;
442a8bd192aSAndreas Gohr        $this->doc .= $lang['apostrophe'];
44357d757d1SAndreas Gohr    }
44457d757d1SAndreas Gohr
4450cecf9d5Sandi    function doublequoteopening() {
44671b40da2SAnika Henke        global $lang;
44771b40da2SAnika Henke        $this->doc .= $lang['doublequoteopening'];
4480cecf9d5Sandi    }
4490cecf9d5Sandi
4500cecf9d5Sandi    function doublequoteclosing() {
45171b40da2SAnika Henke        global $lang;
45271b40da2SAnika Henke        $this->doc .= $lang['doublequoteclosing'];
4530cecf9d5Sandi    }
4540cecf9d5Sandi
4550cecf9d5Sandi    /**
4560cecf9d5Sandi    */
4570cecf9d5Sandi    function camelcaselink($link) {
45811d0aa47Sandi      $this->internallink($link,$link);
4590cecf9d5Sandi    }
4600cecf9d5Sandi
4610b7c14c2Sandi
4620b7c14c2Sandi    function locallink($hash, $name = NULL){
4630b7c14c2Sandi        global $ID;
4640b7c14c2Sandi        $name  = $this->_getLinkTitle($name, $hash, $isImage);
4650b7c14c2Sandi        $hash  = $this->_headerToLink($hash);
4660b7c14c2Sandi        $title = $ID.' &crarr;';
4670b7c14c2Sandi        $this->doc .= '<a href="#'.$hash.'" title="'.$title.'" class="wikilink1">';
4680b7c14c2Sandi        $this->doc .= $name;
4690b7c14c2Sandi        $this->doc .= '</a>';
4700b7c14c2Sandi    }
4710b7c14c2Sandi
472cffcc403Sandi    /**
4733fd0b676Sandi     * Render an internal Wiki Link
4743fd0b676Sandi     *
475*fe9ec250SChris Smith     * $search,$returnonly & $linktype are not for the renderer but are used
476cffcc403Sandi     * elsewhere - no need to implement them in other renderers
4773fd0b676Sandi     *
4783fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
479cffcc403Sandi     */
480*fe9ec250SChris Smith    function internallink($id, $name = NULL, $search=NULL,$returnonly=false,$linktype='content') {
481ba11bd29Sandi        global $conf;
48237e34a5eSandi        global $ID;
4830339c872Sjan        // default name is based on $id as given
4840339c872Sjan        $default = $this->_simpleTitle($id);
485ad32e47eSAndreas Gohr
4860339c872Sjan        // now first resolve and clean up the $id
48737e34a5eSandi        resolve_pageid(getNS($ID),$id,$exists);
488*fe9ec250SChris Smith        $name = $this->_getLinkTitle($name, $default, $isImage, $id, $linktype);
4890e1c636eSandi        if ( !$isImage ) {
4900e1c636eSandi            if ( $exists ) {
491ba11bd29Sandi                $class='wikilink1';
4920cecf9d5Sandi            } else {
493ba11bd29Sandi                $class='wikilink2';
49444a6b4c7SAndreas Gohr                $link['rel']='nofollow';
4950cecf9d5Sandi            }
4960cecf9d5Sandi        } else {
497ba11bd29Sandi            $class='media';
4980cecf9d5Sandi        }
4990cecf9d5Sandi
500a1685bedSandi        //keep hash anchor
501ce6b63d9Schris        list($id,$hash) = explode('#',$id,2);
502943dedc6SAndreas Gohr        if(!empty($hash)) $hash = $this->_headerToLink($hash);
503a1685bedSandi
504ba11bd29Sandi        //prepare for formating
505ba11bd29Sandi        $link['target'] = $conf['target']['wiki'];
506ba11bd29Sandi        $link['style']  = '';
507ba11bd29Sandi        $link['pre']    = '';
508ba11bd29Sandi        $link['suf']    = '';
50940eb54bbSjan        // highlight link to current page
51040eb54bbSjan        if ($id == $ID) {
51192795d04Sandi            $link['pre']    = '<span class="curid">';
51292795d04Sandi            $link['suf']    = '</span>';
51340eb54bbSjan        }
5145e163278SAndreas Gohr        $link['more']   = '';
515ba11bd29Sandi        $link['class']  = $class;
516ba11bd29Sandi        $link['url']    = wl($id);
517ba11bd29Sandi        $link['name']   = $name;
518ba11bd29Sandi        $link['title']  = $id;
519723d78dbSandi        //add search string
520723d78dbSandi        if($search){
521546d3a99SAndreas Gohr            ($conf['userewrite']) ? $link['url'].='?' : $link['url'].='&amp;';
522546d3a99SAndreas Gohr            if(is_array($search)){
523546d3a99SAndreas Gohr                $search = array_map('rawurlencode',$search);
524546d3a99SAndreas Gohr                $link['url'] .= 's[]='.join('&amp;s[]=',$search);
525546d3a99SAndreas Gohr            }else{
526546d3a99SAndreas Gohr                $link['url'] .= 's='.rawurlencode($search);
527546d3a99SAndreas Gohr            }
528723d78dbSandi        }
529723d78dbSandi
530a1685bedSandi        //keep hash
531a1685bedSandi        if($hash) $link['url'].='#'.$hash;
532a1685bedSandi
533ba11bd29Sandi        //output formatted
534cffcc403Sandi        if($returnonly){
535cffcc403Sandi            return $this->_formatLink($link);
536cffcc403Sandi        }else{
537a2d649c4Sandi            $this->doc .= $this->_formatLink($link);
5380cecf9d5Sandi        }
539cffcc403Sandi    }
5400cecf9d5Sandi
541b625487dSandi    function externallink($url, $name = NULL) {
542b625487dSandi        global $conf;
5430cecf9d5Sandi
544433bef32Sandi        $name = $this->_getLinkTitle($name, $url, $isImage);
5456f0c5dbfSandi
5460cecf9d5Sandi        if ( !$isImage ) {
547b625487dSandi            $class='urlextern';
5480cecf9d5Sandi        } else {
549b625487dSandi            $class='media';
5500cecf9d5Sandi        }
5510cecf9d5Sandi
552b625487dSandi        //prepare for formating
553b625487dSandi        $link['target'] = $conf['target']['extern'];
554b625487dSandi        $link['style']  = '';
555b625487dSandi        $link['pre']    = '';
556b625487dSandi        $link['suf']    = '';
5575e163278SAndreas Gohr        $link['more']   = '';
558b625487dSandi        $link['class']  = $class;
559b625487dSandi        $link['url']    = $url;
560e1c10e4dSchris
561b625487dSandi        $link['name']   = $name;
562433bef32Sandi        $link['title']  = $this->_xmlEntities($url);
563b625487dSandi        if($conf['relnofollow']) $link['more'] .= ' rel="nofollow"';
5640cecf9d5Sandi
565b625487dSandi        //output formatted
566a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
5670cecf9d5Sandi    }
5680cecf9d5Sandi
5690cecf9d5Sandi    /**
5700cecf9d5Sandi    */
57197a3e4e3Sandi    function interwikilink($match, $name = NULL, $wikiName, $wikiUri) {
572b625487dSandi        global $conf;
5730cecf9d5Sandi
57497a3e4e3Sandi        $link = array();
57597a3e4e3Sandi        $link['target'] = $conf['target']['interwiki'];
57697a3e4e3Sandi        $link['pre']    = '';
57797a3e4e3Sandi        $link['suf']    = '';
5785e163278SAndreas Gohr        $link['more']   = '';
579433bef32Sandi        $link['name']   = $this->_getLinkTitle($name, $wikiUri, $isImage);
5800cecf9d5Sandi
58197a3e4e3Sandi        //get interwiki URL
5821f82fabeSAndreas Gohr        $url = $this->_resolveInterWiki($wikiName,$wikiUri);
5830cecf9d5Sandi
58497a3e4e3Sandi        if ( !$isImage ) {
5859d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$wikiName);
5869d2ddea4SAndreas Gohr            $link['class'] = "interwiki iw_$class";
5871c2d1019SAndreas Gohr        } else {
5881c2d1019SAndreas Gohr            $link['class'] = 'media';
58997a3e4e3Sandi        }
5900cecf9d5Sandi
59197a3e4e3Sandi        //do we stay at the same server? Use local target
59297a3e4e3Sandi        if( strpos($url,DOKU_URL) === 0 ){
59397a3e4e3Sandi            $link['target'] = $conf['target']['wiki'];
59497a3e4e3Sandi        }
5950cecf9d5Sandi
59697a3e4e3Sandi        $link['url'] = $url;
59797a3e4e3Sandi        $link['title'] = htmlspecialchars($link['url']);
59897a3e4e3Sandi
59997a3e4e3Sandi        //output formatted
600a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6010cecf9d5Sandi    }
6020cecf9d5Sandi
6030cecf9d5Sandi    /**
6040cecf9d5Sandi     */
6051d47afe1Sandi    function windowssharelink($url, $name = NULL) {
6061d47afe1Sandi        global $conf;
6071d47afe1Sandi        global $lang;
6081d47afe1Sandi        //simple setup
6091d47afe1Sandi        $link['target'] = $conf['target']['windows'];
6101d47afe1Sandi        $link['pre']    = '';
6111d47afe1Sandi        $link['suf']   = '';
6121d47afe1Sandi        $link['style']  = '';
6130cecf9d5Sandi
614433bef32Sandi        $link['name'] = $this->_getLinkTitle($name, $url, $isImage);
6150cecf9d5Sandi        if ( !$isImage ) {
6161d47afe1Sandi            $link['class'] = 'windows';
6170cecf9d5Sandi        } else {
6181d47afe1Sandi            $link['class'] = 'media';
6190cecf9d5Sandi        }
6200cecf9d5Sandi
6210cecf9d5Sandi
622433bef32Sandi        $link['title'] = $this->_xmlEntities($url);
6231d47afe1Sandi        $url = str_replace('\\','/',$url);
6241d47afe1Sandi        $url = 'file:///'.$url;
6251d47afe1Sandi        $link['url'] = $url;
6260cecf9d5Sandi
6271d47afe1Sandi        //output formatted
628a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6290cecf9d5Sandi    }
6300cecf9d5Sandi
63171352defSandi    function emaillink($address, $name = NULL) {
63271352defSandi        global $conf;
63371352defSandi        //simple setup
63471352defSandi        $link = array();
63571352defSandi        $link['target'] = '';
63671352defSandi        $link['pre']    = '';
63771352defSandi        $link['suf']   = '';
63871352defSandi        $link['style']  = '';
63971352defSandi        $link['more']   = '';
6400cecf9d5Sandi
641704fb054SAnika Henke        $name = $this->_getLinkTitle($name, $address, $isImage);
6420cecf9d5Sandi        if ( !$isImage ) {
643776b36ecSAndreas Gohr            $link['class']='mail JSnocheck';
6440cecf9d5Sandi        } else {
645776b36ecSAndreas Gohr            $link['class']='media JSnocheck';
6460cecf9d5Sandi        }
6470cecf9d5Sandi
64807738714SAndreas Gohr        $address = $this->_xmlEntities($address);
64900a7b5adSEsther Brunner        $address = obfuscate($address);
65000a7b5adSEsther Brunner        $title   = $address;
6518c128049SAndreas Gohr
65271352defSandi        if(empty($name)){
65300a7b5adSEsther Brunner            $name = $address;
65471352defSandi        }
6558c128049SAndreas Gohr#elseif($isImage{
6568c128049SAndreas Gohr#            $name = $this->_xmlEntities($name);
6578c128049SAndreas Gohr#        }
6580cecf9d5Sandi
659776b36ecSAndreas Gohr        if($conf['mailguard'] == 'visible') $address = rawurlencode($address);
660776b36ecSAndreas Gohr
661776b36ecSAndreas Gohr        $link['url']   = 'mailto:'.$address;
66271352defSandi        $link['name']  = $name;
66371352defSandi        $link['title'] = $title;
6640cecf9d5Sandi
66571352defSandi        //output formatted
666a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6670cecf9d5Sandi    }
6680cecf9d5Sandi
6694826ab45Sandi    function internalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
670dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
67137e34a5eSandi        global $ID;
67237e34a5eSandi        resolve_mediaid(getNS($ID),$src, $exists);
6730cecf9d5Sandi
674d98d4540SBen Coburn        $noLink = false;
6758acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
676b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
6773685f775Sandi
67855efc227SAndreas Gohr        list($ext,$mime) = mimetype($src);
679b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
680dc673a5bSjoe.lapp            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),($linking=='direct'));
6811c882ba8SAndreas Gohr        }elseif($mime == 'application/x-shockwave-flash' && $render){
6822ca14335SEsther Brunner            // don't link flash movies
68344881bd0Shenning.noren            $noLink = true;
68455efc227SAndreas Gohr        }else{
6852ca14335SEsther Brunner            // add file icons
6869d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
6879d2ddea4SAndreas Gohr            $link['class'] .= ' mediafile mf_'.$class;
6886de3759aSAndreas Gohr            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),true);
68955efc227SAndreas Gohr        }
6903685f775Sandi
6916fe20453SGina Haeussge        //markup non existing files
6926fe20453SGina Haeussge        if (!$exists)
6936fe20453SGina Haeussge          $link['class'] .= ' wikilink2';
6946fe20453SGina Haeussge
6953685f775Sandi        //output formatted
696dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
6972ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
6980cecf9d5Sandi    }
6990cecf9d5Sandi
7004826ab45Sandi    function externalmedia ($src, $title=NULL, $align=NULL, $width=NULL,
701dc673a5bSjoe.lapp                            $height=NULL, $cache=NULL, $linking=NULL) {
702d98d4540SBen Coburn        $noLink = false;
7038acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
704b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
705b739ff0fSPierre Spring
706b739ff0fSPierre Spring        $link['url']    = ml($src,array('cache'=>$cache));
7073685f775Sandi
7082ca14335SEsther Brunner        list($ext,$mime) = mimetype($src);
709b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
7102ca14335SEsther Brunner             // link only jpeg images
71144881bd0Shenning.noren             // if ($ext != 'jpg' && $ext != 'jpeg') $noLink = true;
7121c882ba8SAndreas Gohr        }elseif($mime == 'application/x-shockwave-flash' && $render){
7132ca14335SEsther Brunner             // don't link flash movies
71444881bd0Shenning.noren             $noLink = true;
7152ca14335SEsther Brunner        }else{
7162ca14335SEsther Brunner             // add file icons
717d15166e5SAndreas Gohr             $link['class'] .= ' mediafile mf_'.$ext;
7182ca14335SEsther Brunner         }
7192ca14335SEsther Brunner
7203685f775Sandi        //output formatted
721dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
7222ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
7230cecf9d5Sandi    }
7240cecf9d5Sandi
7254826ab45Sandi    /**
7263db95becSAndreas Gohr     * Renders an RSS feed
727b625487dSandi     *
728b625487dSandi     * @author Andreas Gohr <andi@splitbrain.org>
729b625487dSandi     */
7303db95becSAndreas Gohr    function rss ($url,$params){
731b625487dSandi        global $lang;
7323db95becSAndreas Gohr        global $conf;
7333db95becSAndreas Gohr
7343db95becSAndreas Gohr        require_once(DOKU_INC.'inc/FeedParser.php');
7353db95becSAndreas Gohr        $feed = new FeedParser();
73600077af8SAndreas Gohr        $feed->set_feed_url($url);
737b625487dSandi
738b625487dSandi        //disable warning while fetching
739bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { $elvl = error_reporting(E_ERROR); }
7403db95becSAndreas Gohr        $rc = $feed->init();
741bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { error_reporting($elvl); }
742b625487dSandi
7433db95becSAndreas Gohr        //decide on start and end
7443db95becSAndreas Gohr        if($params['reverse']){
7453db95becSAndreas Gohr            $mod = -1;
7463db95becSAndreas Gohr            $start = $feed->get_item_quantity()-1;
7473db95becSAndreas Gohr            $end   = $start - ($params['max']);
748b2a412b0SAndreas Gohr            $end   = ($end < -1) ? -1 : $end;
7493db95becSAndreas Gohr        }else{
7503db95becSAndreas Gohr            $mod   = 1;
7513db95becSAndreas Gohr            $start = 0;
7523db95becSAndreas Gohr            $end   = $feed->get_item_quantity();
7533db95becSAndreas Gohr            $end   = ($end > $params['max']) ? $params['max'] : $end;;
7543db95becSAndreas Gohr        }
7553db95becSAndreas Gohr
756a2d649c4Sandi        $this->doc .= '<ul class="rss">';
7573db95becSAndreas Gohr        if($rc){
7583db95becSAndreas Gohr            for ($x = $start; $x != $end; $x += $mod) {
7591bde1582SAndreas Gohr                $item = $feed->get_item($x);
7603db95becSAndreas Gohr                $this->doc .= '<li><div class="li">';
761d2ea3363SAndreas Gohr                // support feeds without links
762d2ea3363SAndreas Gohr                $lnkurl = $item->get_permalink();
763d2ea3363SAndreas Gohr                if($lnkurl){
7641bde1582SAndreas Gohr                    $this->externallink($item->get_permalink(),
7651bde1582SAndreas Gohr                                        $item->get_title());
766d2ea3363SAndreas Gohr                }else{
767d2ea3363SAndreas Gohr                    $this->doc .= ' '.$item->get_title();
768d2ea3363SAndreas Gohr                }
7693db95becSAndreas Gohr                if($params['author']){
7701bde1582SAndreas Gohr                    $author = $item->get_author(0);
7711bde1582SAndreas Gohr                    if($author){
7721bde1582SAndreas Gohr                        $name = $author->get_name();
7731bde1582SAndreas Gohr                        if(!$name) $name = $author->get_email();
7741bde1582SAndreas Gohr                        if($name) $this->doc .= ' '.$lang['by'].' '.$name;
7751bde1582SAndreas Gohr                    }
7763db95becSAndreas Gohr                }
7773db95becSAndreas Gohr                if($params['date']){
7782e7e0c29SAndreas Gohr                    $this->doc .= ' ('.$item->get_local_date($conf['dformat']).')';
7793db95becSAndreas Gohr                }
7801bde1582SAndreas Gohr                if($params['details']){
7813db95becSAndreas Gohr                    $this->doc .= '<div class="detail">';
782173dccb7STom N Harris                    if($conf['htmlok']){
7831bde1582SAndreas Gohr                        $this->doc .= $item->get_description();
7843db95becSAndreas Gohr                    }else{
7851bde1582SAndreas Gohr                        $this->doc .= strip_tags($item->get_description());
7863db95becSAndreas Gohr                    }
7873db95becSAndreas Gohr                    $this->doc .= '</div>';
7883db95becSAndreas Gohr                }
7893db95becSAndreas Gohr
7903db95becSAndreas Gohr                $this->doc .= '</div></li>';
791b625487dSandi            }
792b625487dSandi        }else{
7933db95becSAndreas Gohr            $this->doc .= '<li><div class="li">';
794a2d649c4Sandi            $this->doc .= '<em>'.$lang['rssfailed'].'</em>';
795b625487dSandi            $this->externallink($url);
79645e147ccSAndreas Gohr            if($conf['allowdebug']){
79745e147ccSAndreas Gohr                $this->doc .= '<!--'.hsc($feed->error).'-->';
79845e147ccSAndreas Gohr            }
7993db95becSAndreas Gohr            $this->doc .= '</div></li>';
800b625487dSandi        }
801a2d649c4Sandi        $this->doc .= '</ul>';
802b625487dSandi    }
803b625487dSandi
8040cecf9d5Sandi    // $numrows not yet implemented
8050cecf9d5Sandi    function table_open($maxcols = NULL, $numrows = NULL){
806b5742cedSPierre Spring        // initialize the row counter used for classes
807b5742cedSPierre Spring        $this->_counter['row_counter'] = 0;
808a2d649c4Sandi        $this->doc .= '<table class="inline">'.DOKU_LF;
8090cecf9d5Sandi    }
8100cecf9d5Sandi
8110cecf9d5Sandi    function table_close(){
812cb42b03dSAnika Henke        $this->doc .= '</table>'.DOKU_LF;
8130cecf9d5Sandi    }
8140cecf9d5Sandi
8150cecf9d5Sandi    function tablerow_open(){
816b5742cedSPierre Spring        // initialize the cell counter used for classes
817b5742cedSPierre Spring        $this->_counter['cell_counter'] = 0;
818b5742cedSPierre Spring        $class = 'row' . $this->_counter['row_counter']++;
819b5742cedSPierre Spring        $this->doc .= DOKU_TAB . '<tr class="'.$class.'">' . DOKU_LF . DOKU_TAB . DOKU_TAB;
8200cecf9d5Sandi    }
8210cecf9d5Sandi
8220cecf9d5Sandi    function tablerow_close(){
823a2d649c4Sandi        $this->doc .= DOKU_LF . DOKU_TAB . '</tr>' . DOKU_LF;
8240cecf9d5Sandi    }
8250cecf9d5Sandi
8260cecf9d5Sandi    function tableheader_open($colspan = 1, $align = NULL){
827b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
8280cecf9d5Sandi        if ( !is_null($align) ) {
829b5742cedSPierre Spring            $class .= ' '.$align.'align';
8300cecf9d5Sandi        }
831b5742cedSPierre Spring        $class .= '"';
832b5742cedSPierre Spring        $this->doc .= '<th ' . $class;
8330cecf9d5Sandi        if ( $colspan > 1 ) {
834a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
835a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8360cecf9d5Sandi        }
837a2d649c4Sandi        $this->doc .= '>';
8380cecf9d5Sandi    }
8390cecf9d5Sandi
8400cecf9d5Sandi    function tableheader_close(){
841a2d649c4Sandi        $this->doc .= '</th>';
8420cecf9d5Sandi    }
8430cecf9d5Sandi
8440cecf9d5Sandi    function tablecell_open($colspan = 1, $align = NULL){
845b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
8460cecf9d5Sandi        if ( !is_null($align) ) {
847b5742cedSPierre Spring            $class .= ' '.$align.'align';
8480cecf9d5Sandi        }
849b5742cedSPierre Spring        $class .= '"';
850b5742cedSPierre Spring        $this->doc .= '<td '.$class;
8510cecf9d5Sandi        if ( $colspan > 1 ) {
852a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
853a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
8540cecf9d5Sandi        }
855a2d649c4Sandi        $this->doc .= '>';
8560cecf9d5Sandi    }
8570cecf9d5Sandi
8580cecf9d5Sandi    function tablecell_close(){
859a2d649c4Sandi        $this->doc .= '</td>';
8600cecf9d5Sandi    }
8610cecf9d5Sandi
8620cecf9d5Sandi    //----------------------------------------------------------
8630cecf9d5Sandi    // Utils
8640cecf9d5Sandi
865ba11bd29Sandi    /**
8663fd0b676Sandi     * Build a link
8673fd0b676Sandi     *
8683fd0b676Sandi     * Assembles all parts defined in $link returns HTML for the link
869ba11bd29Sandi     *
870ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
871ba11bd29Sandi     */
872433bef32Sandi    function _formatLink($link){
873ba11bd29Sandi        //make sure the url is XHTML compliant (skip mailto)
874ba11bd29Sandi        if(substr($link['url'],0,7) != 'mailto:'){
875ba11bd29Sandi            $link['url'] = str_replace('&','&amp;',$link['url']);
876ba11bd29Sandi            $link['url'] = str_replace('&amp;amp;','&amp;',$link['url']);
877ba11bd29Sandi        }
878ba11bd29Sandi        //remove double encodings in titles
879ba11bd29Sandi        $link['title'] = str_replace('&amp;amp;','&amp;',$link['title']);
880ba11bd29Sandi
881453493f2SAndreas Gohr        // be sure there are no bad chars in url or title
882453493f2SAndreas Gohr        // (we can't do this for name because it can contain an img tag)
883453493f2SAndreas Gohr        $link['url']   = strtr($link['url'],array('>'=>'%3E','<'=>'%3C','"'=>'%22'));
884453493f2SAndreas Gohr        $link['title'] = strtr($link['title'],array('>'=>'&gt;','<'=>'&lt;','"'=>'&quot;'));
885453493f2SAndreas Gohr
886ba11bd29Sandi        $ret  = '';
887ba11bd29Sandi        $ret .= $link['pre'];
888ba11bd29Sandi        $ret .= '<a href="'.$link['url'].'"';
889bb4866bdSchris        if(!empty($link['class']))  $ret .= ' class="'.$link['class'].'"';
890bb4866bdSchris        if(!empty($link['target'])) $ret .= ' target="'.$link['target'].'"';
891bb4866bdSchris        if(!empty($link['title']))  $ret .= ' title="'.$link['title'].'"';
892bb4866bdSchris        if(!empty($link['style']))  $ret .= ' style="'.$link['style'].'"';
89344a6b4c7SAndreas Gohr        if(!empty($link['rel']))    $ret .= ' rel="'.$link['rel'].'"';
894bb4866bdSchris        if(!empty($link['more']))   $ret .= ' '.$link['more'];
895ba11bd29Sandi        $ret .= '>';
896ba11bd29Sandi        $ret .= $link['name'];
897ba11bd29Sandi        $ret .= '</a>';
898ba11bd29Sandi        $ret .= $link['suf'];
899ba11bd29Sandi        return $ret;
900ba11bd29Sandi    }
901ba11bd29Sandi
902ba11bd29Sandi    /**
9033fd0b676Sandi     * Renders internal and external media
9043fd0b676Sandi     *
9053fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
9063fd0b676Sandi     */
9073fd0b676Sandi    function _media ($src, $title=NULL, $align=NULL, $width=NULL,
908b739ff0fSPierre Spring                      $height=NULL, $cache=NULL, $render = true) {
9093fd0b676Sandi
9103fd0b676Sandi        $ret = '';
9113fd0b676Sandi
9123fd0b676Sandi        list($ext,$mime) = mimetype($src);
9133fd0b676Sandi        if(substr($mime,0,5) == 'image'){
914b739ff0fSPierre Spring            // first get the $title
915b739ff0fSPierre Spring            if (!is_null($title)) {
916b739ff0fSPierre Spring                $title  = $this->_xmlEntities($title);
917b739ff0fSPierre Spring            }elseif($ext == 'jpg' || $ext == 'jpeg'){
918b739ff0fSPierre Spring                //try to use the caption from IPTC/EXIF
919b739ff0fSPierre Spring                require_once(DOKU_INC.'inc/JpegMeta.php');
920b739ff0fSPierre Spring                $jpeg =& new JpegMeta(mediaFN($src));
921b739ff0fSPierre Spring                if($jpeg !== false) $cap = $jpeg->getTitle();
922b739ff0fSPierre Spring                if($cap){
923b739ff0fSPierre Spring                    $title = $this->_xmlEntities($cap);
924b739ff0fSPierre Spring                }
925b739ff0fSPierre Spring            }
926b739ff0fSPierre Spring            if (!$render) {
927b739ff0fSPierre Spring                // if the picture is not supposed to be rendered
928b739ff0fSPierre Spring                // return the title of the picture
929b739ff0fSPierre Spring                if (!$title) {
930b739ff0fSPierre Spring                    // just show the sourcename
931b739ff0fSPierre Spring                    $title = $this->_xmlEntities(basename(noNS($src)));
932b739ff0fSPierre Spring                }
933b739ff0fSPierre Spring                return $title;
934b739ff0fSPierre Spring            }
9353fd0b676Sandi            //add image tag
9366de3759aSAndreas Gohr            $ret .= '<img src="'.ml($src,array('w'=>$width,'h'=>$height,'cache'=>$cache)).'"';
9373fd0b676Sandi            $ret .= ' class="media'.$align.'"';
9383fd0b676Sandi
9394ab889eaSAndreas Gohr            // make left/right alignment for no-CSS view work (feeds)
9404ab889eaSAndreas Gohr            if($align == 'right') $ret .= ' align="right"';
9414ab889eaSAndreas Gohr            if($align == 'left')  $ret .= ' align="left"';
9424ab889eaSAndreas Gohr
943b739ff0fSPierre Spring            if ($title) {
944b739ff0fSPierre Spring                $ret .= ' title="' . $title . '"';
945b739ff0fSPierre Spring                $ret .= ' alt="'   . $title .'"';
9463fd0b676Sandi            }else{
9473fd0b676Sandi                $ret .= ' alt=""';
9483fd0b676Sandi            }
9493fd0b676Sandi
9503fd0b676Sandi            if ( !is_null($width) )
9513fd0b676Sandi                $ret .= ' width="'.$this->_xmlEntities($width).'"';
9523fd0b676Sandi
9533fd0b676Sandi            if ( !is_null($height) )
9543fd0b676Sandi                $ret .= ' height="'.$this->_xmlEntities($height).'"';
9553fd0b676Sandi
9563fd0b676Sandi            $ret .= ' />';
9573fd0b676Sandi
9583fd0b676Sandi        }elseif($mime == 'application/x-shockwave-flash'){
9591c882ba8SAndreas Gohr            if (!$render) {
9601c882ba8SAndreas Gohr                // if the flash is not supposed to be rendered
9611c882ba8SAndreas Gohr                // return the title of the flash
9621c882ba8SAndreas Gohr                if (!$title) {
9631c882ba8SAndreas Gohr                    // just show the sourcename
9641c882ba8SAndreas Gohr                    $title = $this->_xmlEntities(basename(noNS($src)));
9651c882ba8SAndreas Gohr                }
9661c882ba8SAndreas Gohr                return $title;
9671c882ba8SAndreas Gohr            }
9681c882ba8SAndreas Gohr
9693fd0b676Sandi            $ret .= '<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"'.
9703fd0b676Sandi                    ' codebase="http://download.macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version=6,0,40,0"';
9713fd0b676Sandi            if ( !is_null($width) ) $ret .= ' width="'.$this->_xmlEntities($width).'"';
9723fd0b676Sandi            if ( !is_null($height) ) $ret .= ' height="'.$this->_xmlEntities($height).'"';
9733fd0b676Sandi            $ret .= '>'.DOKU_LF;
9746de3759aSAndreas Gohr            $ret .= '<param name="movie" value="'.ml($src).'" />'.DOKU_LF;
9753fd0b676Sandi            $ret .= '<param name="quality" value="high" />'.DOKU_LF;
9766de3759aSAndreas Gohr            $ret .= '<embed src="'.ml($src).'"'.
9773fd0b676Sandi                    ' quality="high"';
9783fd0b676Sandi            if ( !is_null($width) ) $ret .= ' width="'.$this->_xmlEntities($width).'"';
9793fd0b676Sandi            if ( !is_null($height) ) $ret .= ' height="'.$this->_xmlEntities($height).'"';
9803fd0b676Sandi            $ret .= ' type="application/x-shockwave-flash"'.
9813fd0b676Sandi                    ' pluginspage="http://www.macromedia.com/go/getflashplayer"></embed>'.DOKU_LF;
9823fd0b676Sandi            $ret .= '</object>'.DOKU_LF;
9833fd0b676Sandi
9840f428d7dSAndreas Gohr        }elseif($title){
9853fd0b676Sandi            // well at least we have a title to display
9863fd0b676Sandi            $ret .= $this->_xmlEntities($title);
9873fd0b676Sandi        }else{
9885291ca3aSAndreas Gohr            // just show the sourcename
9890f428d7dSAndreas Gohr            $ret .= $this->_xmlEntities(basename(noNS($src)));
9903fd0b676Sandi        }
9913fd0b676Sandi
9923fd0b676Sandi        return $ret;
9933fd0b676Sandi    }
9943fd0b676Sandi
995433bef32Sandi    function _xmlEntities($string) {
996de117061Schris        return htmlspecialchars($string,ENT_QUOTES,'UTF-8');
9970cecf9d5Sandi    }
9980cecf9d5Sandi
9998a831f2bSAndreas Gohr    /**
10008a831f2bSAndreas Gohr     * Creates a linkid from a headline
1001c5a8fd96SAndreas Gohr     *
1002c5a8fd96SAndreas Gohr     * @param string  $title   The headline title
1003c5a8fd96SAndreas Gohr     * @param boolean $create  Create a new unique ID?
1004c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
10058a831f2bSAndreas Gohr     */
1006c5a8fd96SAndreas Gohr    function _headerToLink($title,$create=false) {
1007501af51eSAndreas Gohr        $title = str_replace(':','',cleanID($title));
1008a9e0fd1bSAndreas Gohr        $title = ltrim($title,'0123456789._-');
10098a831f2bSAndreas Gohr        if(empty($title)) $title='section';
1010c5a8fd96SAndreas Gohr
1011c5a8fd96SAndreas Gohr        if($create){
1012c5a8fd96SAndreas Gohr            // make sure tiles are unique
1013c5a8fd96SAndreas Gohr            $num = '';
1014c5a8fd96SAndreas Gohr            while(in_array($title.$num,$this->headers)){
10150affd488SAndreas Gohr                ($num) ? $num++ : $num = 1;
1016c5a8fd96SAndreas Gohr            }
1017c5a8fd96SAndreas Gohr            $title = $title.$num;
1018c5a8fd96SAndreas Gohr            $this->headers[] = $title;
1019c5a8fd96SAndreas Gohr        }
1020c5a8fd96SAndreas Gohr
10218a831f2bSAndreas Gohr        return $title;
10220cecf9d5Sandi    }
10230cecf9d5Sandi
1024af587fa8Sandi    /**
10253fd0b676Sandi     * Construct a title and handle images in titles
10263fd0b676Sandi     *
10270b7c14c2Sandi     * @author Harry Fuecks <hfuecks@gmail.com>
10283fd0b676Sandi     */
1029*fe9ec250SChris Smith    function _getLinkTitle($title, $default, & $isImage, $id=NULL, $linktype='content') {
1030bb0a59d4Sjan        global $conf;
1031bb0a59d4Sjan
103244881bd0Shenning.noren        $isImage = false;
1033a6f3bd20SAnika Henke        if ( is_null($title) || trim($title)=='') {
1034*fe9ec250SChris Smith            if (useHeading($linktype) && $id) {
1035fc18c0fbSchris                $heading = p_get_first_heading($id,true);
1036bb0a59d4Sjan                if ($heading) {
1037433bef32Sandi                    return $this->_xmlEntities($heading);
1038bb0a59d4Sjan                }
1039bb0a59d4Sjan            }
1040433bef32Sandi            return $this->_xmlEntities($default);
10410cecf9d5Sandi        } else if ( is_array($title) ) {
104244881bd0Shenning.noren            $isImage = true;
1043433bef32Sandi            return $this->_imageTitle($title);
104468c26e6dSMichael Klier        } else {
104568c26e6dSMichael Klier            return $this->_xmlEntities($title);
10460cecf9d5Sandi        }
10470cecf9d5Sandi    }
10480cecf9d5Sandi
10490cecf9d5Sandi    /**
10503fd0b676Sandi     * Returns an HTML code for images used in link titles
10513fd0b676Sandi     *
10523fd0b676Sandi     * @todo Resolve namespace on internal images
10533fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
10540cecf9d5Sandi     */
1055433bef32Sandi    function _imageTitle($img) {
1056433bef32Sandi        return $this->_media($img['src'],
10574826ab45Sandi                              $img['title'],
10584826ab45Sandi                              $img['align'],
10594826ab45Sandi                              $img['width'],
10604826ab45Sandi                              $img['height'],
10614826ab45Sandi                              $img['cache']);
10620cecf9d5Sandi    }
1063b739ff0fSPierre Spring
1064b739ff0fSPierre Spring    /**
1065b739ff0fSPierre Spring     * _getMediaLinkConf is a helperfunction to internalmedia() and externalmedia()
1066b739ff0fSPierre Spring     * which returns a basic link to a media.
1067b739ff0fSPierre Spring     *
1068b739ff0fSPierre Spring     * @author Pierre Spring <pierre.spring@liip.ch>
1069b739ff0fSPierre Spring     * @param string $src
1070b739ff0fSPierre Spring     * @param string $title
1071b739ff0fSPierre Spring     * @param string $align
1072b739ff0fSPierre Spring     * @param string $width
1073b739ff0fSPierre Spring     * @param string $height
1074b739ff0fSPierre Spring     * @param string $cache
1075b739ff0fSPierre Spring     * @param string $render
1076b739ff0fSPierre Spring     * @access protected
1077b739ff0fSPierre Spring     * @return array
1078b739ff0fSPierre Spring     */
1079b739ff0fSPierre Spring    function _getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render)
1080b739ff0fSPierre Spring    {
1081b739ff0fSPierre Spring        global $conf;
1082b739ff0fSPierre Spring
1083b739ff0fSPierre Spring        $link = array();
1084b739ff0fSPierre Spring        $link['class']  = 'media';
1085b739ff0fSPierre Spring        $link['style']  = '';
1086b739ff0fSPierre Spring        $link['pre']    = '';
1087b739ff0fSPierre Spring        $link['suf']    = '';
1088b739ff0fSPierre Spring        $link['more']   = '';
1089b739ff0fSPierre Spring        $link['target'] = $conf['target']['media'];
1090b739ff0fSPierre Spring        $link['title']  = $this->_xmlEntities($src);
1091b739ff0fSPierre Spring        $link['name']   = $this->_media($src, $title, $align, $width, $height, $cache, $render);
1092b739ff0fSPierre Spring
1093b739ff0fSPierre Spring        return $link;
1094b739ff0fSPierre Spring    }
10950cecf9d5Sandi}
10960cecf9d5Sandi
10974826ab45Sandi//Setup VIM: ex: et ts=4 enc=utf-8 :
1098