xref: /dokuwiki/inc/parser/xhtml.php (revision f05a1cc5fcdb4c2b6ee3cbf499f980f800dbd105)
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
32fb838798SDanny    var $sectionedits = array(); // A stack of section edit data
33b04a190dSMichael Hamann    private $lastsecid = 0; // last section edit id, used by startSectionEdit
340cecf9d5Sandi
350cecf9d5Sandi    var $headers = array();
3616ec3e37SAndreas Gohr    /** @var array a list of footnotes, list starts at 1! */
370cecf9d5Sandi    var $footnotes = array();
3891459163SAnika Henke    var $lastlevel = 0;
3991459163SAnika Henke    var $node = array(0,0,0,0,0);
407764a90aSandi    var $store = '';
417764a90aSandi
42b5742cedSPierre Spring    var $_counter   = array(); // used as global counter, introduced for table classes
433d491f75SAndreas Gohr    var $_codeblock = 0; // counts the code and file blocks, used to provide download links
44b5742cedSPierre Spring
4590df9a4dSAdrian Lang    /**
4690df9a4dSAdrian Lang     * Register a new edit section range
4790df9a4dSAdrian Lang     *
4890df9a4dSAdrian Lang     * @param $type  string The section type identifier
4990df9a4dSAdrian Lang     * @param $title string The section title
5090df9a4dSAdrian Lang     * @param $start int    The byte position for the edit start
5190df9a4dSAdrian Lang     * @return string A marker class for the starting HTML element
5290df9a4dSAdrian Lang     * @author Adrian Lang <lang@cosmocode.de>
5390df9a4dSAdrian Lang     */
543f9e3215SAdrian Lang    public function startSectionEdit($start, $type, $title = null) {
55b04a190dSMichael Hamann        $this->sectionedits[] = array(++$this->lastsecid, $start, $type, $title);
56b04a190dSMichael Hamann        return 'sectionedit' . $this->lastsecid;
5790df9a4dSAdrian Lang    }
5890df9a4dSAdrian Lang
5990df9a4dSAdrian Lang    /**
6090df9a4dSAdrian Lang     * Finish an edit section range
6190df9a4dSAdrian Lang     *
62d9e36cbeSAdrian Lang     * @param $end int The byte position for the edit end; null for the rest of
63c404cb3bSMatt Perry     *                 the page
6490df9a4dSAdrian Lang     * @author Adrian Lang <lang@cosmocode.de>
6590df9a4dSAdrian Lang     */
663f9e3215SAdrian Lang    public function finishSectionEdit($end = null) {
6790df9a4dSAdrian Lang        list($id, $start, $type, $title) = array_pop($this->sectionedits);
68d9e36cbeSAdrian Lang        if (!is_null($end) && $end <= $start) {
6900c13053SAdrian Lang            return;
7000c13053SAdrian Lang        }
7140868f2fSAdrian Lang        $this->doc .= "<!-- EDIT$id " . strtoupper($type) . ' ';
7240868f2fSAdrian Lang        if (!is_null($title)) {
7340868f2fSAdrian Lang            $this->doc .= '"' . str_replace('"', '', $title) . '" ';
7440868f2fSAdrian Lang        }
75d9e36cbeSAdrian Lang        $this->doc .= "[$start-" . (is_null($end) ? '' : $end) . '] -->';
7690df9a4dSAdrian Lang    }
7790df9a4dSAdrian Lang
785f70445dSAndreas Gohr    function getFormat(){
795f70445dSAndreas Gohr        return 'xhtml';
805f70445dSAndreas Gohr    }
815f70445dSAndreas Gohr
825f70445dSAndreas Gohr
830cecf9d5Sandi    function document_start() {
84c5a8fd96SAndreas Gohr        //reset some internals
85c5a8fd96SAndreas Gohr        $this->toc     = array();
86c5a8fd96SAndreas Gohr        $this->headers = array();
870cecf9d5Sandi    }
880cecf9d5Sandi
890cecf9d5Sandi    function document_end() {
9090df9a4dSAdrian Lang        // Finish open section edits.
9190df9a4dSAdrian Lang        while (count($this->sectionedits) > 0) {
9290df9a4dSAdrian Lang            if ($this->sectionedits[count($this->sectionedits) - 1][1] <= 1) {
9390df9a4dSAdrian Lang                // If there is only one section, do not write a section edit
9490df9a4dSAdrian Lang                // marker.
9590df9a4dSAdrian Lang                array_pop($this->sectionedits);
9690df9a4dSAdrian Lang            } else {
97d9e36cbeSAdrian Lang                $this->finishSectionEdit();
9890df9a4dSAdrian Lang            }
9990df9a4dSAdrian Lang        }
10090df9a4dSAdrian Lang
1010cecf9d5Sandi        if ( count ($this->footnotes) > 0 ) {
102a2d649c4Sandi            $this->doc .= '<div class="footnotes">'.DOKU_LF;
103d74aace9Schris
10416ec3e37SAndreas Gohr            foreach ( $this->footnotes as $id => $footnote ) {
105d74aace9Schris                // check its not a placeholder that indicates actual footnote text is elsewhere
106d74aace9Schris                if (substr($footnote, 0, 5) != "@@FNT") {
107d74aace9Schris
108d74aace9Schris                    // open the footnote and set the anchor and backlink
109d74aace9Schris                    $this->doc .= '<div class="fn">';
11016cc7ed7SAnika Henke                    $this->doc .= '<sup><a href="#fnt__'.$id.'" id="fn__'.$id.'" class="fn_bot">';
11129bfcd16SAndreas Gohr                    $this->doc .= $id.')</a></sup> '.DOKU_LF;
112d74aace9Schris
113d74aace9Schris                    // get any other footnotes that use the same markup
114d74aace9Schris                    $alt = array_keys($this->footnotes, "@@FNT$id");
115d74aace9Schris
116d74aace9Schris                    if (count($alt)) {
117d74aace9Schris                        foreach ($alt as $ref) {
118d74aace9Schris                            // set anchor and backlink for the other footnotes
11916ec3e37SAndreas Gohr                            $this->doc .= ', <sup><a href="#fnt__'.($ref).'" id="fn__'.($ref).'" class="fn_bot">';
12016ec3e37SAndreas Gohr                            $this->doc .= ($ref).')</a></sup> '.DOKU_LF;
121d74aace9Schris                        }
122d74aace9Schris                    }
123d74aace9Schris
124d74aace9Schris                    // add footnote markup and close this footnote
125a2d649c4Sandi                    $this->doc .= $footnote;
126d74aace9Schris                    $this->doc .= '</div>' . DOKU_LF;
127d74aace9Schris                }
1280cecf9d5Sandi            }
129a2d649c4Sandi            $this->doc .= '</div>'.DOKU_LF;
1300cecf9d5Sandi        }
131c5a8fd96SAndreas Gohr
132b8595a66SAndreas Gohr        // Prepare the TOC
133851f2e89SAnika Henke        global $conf;
134851f2e89SAnika Henke        if($this->info['toc'] && is_array($this->toc) && $conf['tocminheads'] && count($this->toc) >= $conf['tocminheads']){
135b8595a66SAndreas Gohr            global $TOC;
136b8595a66SAndreas Gohr            $TOC = $this->toc;
1370cecf9d5Sandi        }
1383e55d035SAndreas Gohr
1393e55d035SAndreas Gohr        // make sure there are no empty paragraphs
14027918226Schris        $this->doc = preg_replace('#<p>\s*</p>#','',$this->doc);
141e41c4da9SAndreas Gohr    }
1420cecf9d5Sandi
143e7856beaSchris    function toc_additem($id, $text, $level) {
144af587fa8Sandi        global $conf;
145af587fa8Sandi
146c5a8fd96SAndreas Gohr        //handle TOC
147c5a8fd96SAndreas Gohr        if($level >= $conf['toptoclevel'] && $level <= $conf['maxtoclevel']){
1487d91652aSAndreas Gohr            $this->toc[] = html_mktocitem($id, $text, $level-$conf['toptoclevel']+1);
149c5a8fd96SAndreas Gohr        }
150e7856beaSchris    }
151e7856beaSchris
152e7856beaSchris    function header($text, $level, $pos) {
15390df9a4dSAdrian Lang        global $conf;
15490df9a4dSAdrian Lang
155bdd8111bSAndreas Gohr        if(!$text) return; //skip empty headlines
156e7856beaSchris
157e7856beaSchris        $hid = $this->_headerToLink($text,true);
158e7856beaSchris
159e7856beaSchris        //only add items within configured levels
160e7856beaSchris        $this->toc_additem($hid, $text, $level);
161c5a8fd96SAndreas Gohr
16291459163SAnika Henke        // adjust $node to reflect hierarchy of levels
16391459163SAnika Henke        $this->node[$level-1]++;
16491459163SAnika Henke        if ($level < $this->lastlevel) {
16591459163SAnika Henke            for ($i = 0; $i < $this->lastlevel-$level; $i++) {
16691459163SAnika Henke                $this->node[$this->lastlevel-$i-1] = 0;
16791459163SAnika Henke            }
16891459163SAnika Henke        }
16991459163SAnika Henke        $this->lastlevel = $level;
17091459163SAnika Henke
17190df9a4dSAdrian Lang        if ($level <= $conf['maxseclevel'] &&
17290df9a4dSAdrian Lang            count($this->sectionedits) > 0 &&
17390df9a4dSAdrian Lang            $this->sectionedits[count($this->sectionedits) - 1][2] === 'section') {
1746c1f778cSAdrian Lang            $this->finishSectionEdit($pos - 1);
17590df9a4dSAdrian Lang        }
17690df9a4dSAdrian Lang
177c5a8fd96SAndreas Gohr        // write the header
17890df9a4dSAdrian Lang        $this->doc .= DOKU_LF.'<h'.$level;
17990df9a4dSAdrian Lang        if ($level <= $conf['maxseclevel']) {
18090df9a4dSAdrian Lang            $this->doc .= ' class="' . $this->startSectionEdit($pos, 'section', $text) . '"';
18190df9a4dSAdrian Lang        }
18216cc7ed7SAnika Henke        $this->doc .= ' id="'.$hid.'">';
183a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
18416cc7ed7SAnika Henke        $this->doc .= "</h$level>".DOKU_LF;
1850cecf9d5Sandi    }
1860cecf9d5Sandi
1870cecf9d5Sandi    function section_open($level) {
1889864e7b1SAdrian Lang        $this->doc .= '<div class="level' . $level . '">' . DOKU_LF;
1890cecf9d5Sandi    }
1900cecf9d5Sandi
1910cecf9d5Sandi    function section_close() {
192a2d649c4Sandi        $this->doc .= DOKU_LF.'</div>'.DOKU_LF;
1930cecf9d5Sandi    }
1940cecf9d5Sandi
1950cecf9d5Sandi    function cdata($text) {
196a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
1970cecf9d5Sandi    }
1980cecf9d5Sandi
1990cecf9d5Sandi    function p_open() {
20059869a4bSAnika Henke        $this->doc .= DOKU_LF.'<p>'.DOKU_LF;
2010cecf9d5Sandi    }
2020cecf9d5Sandi
2030cecf9d5Sandi    function p_close() {
20459869a4bSAnika Henke        $this->doc .= DOKU_LF.'</p>'.DOKU_LF;
2050cecf9d5Sandi    }
2060cecf9d5Sandi
2070cecf9d5Sandi    function linebreak() {
208a2d649c4Sandi        $this->doc .= '<br/>'.DOKU_LF;
2090cecf9d5Sandi    }
2100cecf9d5Sandi
2110cecf9d5Sandi    function hr() {
2124beabca9SAnika Henke        $this->doc .= '<hr />'.DOKU_LF;
2130cecf9d5Sandi    }
2140cecf9d5Sandi
2150cecf9d5Sandi    function strong_open() {
216a2d649c4Sandi        $this->doc .= '<strong>';
2170cecf9d5Sandi    }
2180cecf9d5Sandi
2190cecf9d5Sandi    function strong_close() {
220a2d649c4Sandi        $this->doc .= '</strong>';
2210cecf9d5Sandi    }
2220cecf9d5Sandi
2230cecf9d5Sandi    function emphasis_open() {
224a2d649c4Sandi        $this->doc .= '<em>';
2250cecf9d5Sandi    }
2260cecf9d5Sandi
2270cecf9d5Sandi    function emphasis_close() {
228a2d649c4Sandi        $this->doc .= '</em>';
2290cecf9d5Sandi    }
2300cecf9d5Sandi
2310cecf9d5Sandi    function underline_open() {
23202e51121SAnika Henke        $this->doc .= '<em class="u">';
2330cecf9d5Sandi    }
2340cecf9d5Sandi
2350cecf9d5Sandi    function underline_close() {
23602e51121SAnika Henke        $this->doc .= '</em>';
2370cecf9d5Sandi    }
2380cecf9d5Sandi
2390cecf9d5Sandi    function monospace_open() {
240a2d649c4Sandi        $this->doc .= '<code>';
2410cecf9d5Sandi    }
2420cecf9d5Sandi
2430cecf9d5Sandi    function monospace_close() {
244a2d649c4Sandi        $this->doc .= '</code>';
2450cecf9d5Sandi    }
2460cecf9d5Sandi
2470cecf9d5Sandi    function subscript_open() {
248a2d649c4Sandi        $this->doc .= '<sub>';
2490cecf9d5Sandi    }
2500cecf9d5Sandi
2510cecf9d5Sandi    function subscript_close() {
252a2d649c4Sandi        $this->doc .= '</sub>';
2530cecf9d5Sandi    }
2540cecf9d5Sandi
2550cecf9d5Sandi    function superscript_open() {
256a2d649c4Sandi        $this->doc .= '<sup>';
2570cecf9d5Sandi    }
2580cecf9d5Sandi
2590cecf9d5Sandi    function superscript_close() {
260a2d649c4Sandi        $this->doc .= '</sup>';
2610cecf9d5Sandi    }
2620cecf9d5Sandi
2630cecf9d5Sandi    function deleted_open() {
264a2d649c4Sandi        $this->doc .= '<del>';
2650cecf9d5Sandi    }
2660cecf9d5Sandi
2670cecf9d5Sandi    function deleted_close() {
268a2d649c4Sandi        $this->doc .= '</del>';
2690cecf9d5Sandi    }
2700cecf9d5Sandi
2713fd0b676Sandi    /**
2723fd0b676Sandi     * Callback for footnote start syntax
2733fd0b676Sandi     *
2743fd0b676Sandi     * All following content will go to the footnote instead of
275d74aace9Schris     * the document. To achieve this the previous rendered content
2763fd0b676Sandi     * is moved to $store and $doc is cleared
2773fd0b676Sandi     *
2783fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
2793fd0b676Sandi     */
2800cecf9d5Sandi    function footnote_open() {
2817764a90aSandi
2827764a90aSandi        // move current content to store and record footnote
2837764a90aSandi        $this->store = $this->doc;
2847764a90aSandi        $this->doc   = '';
2850cecf9d5Sandi    }
2860cecf9d5Sandi
2873fd0b676Sandi    /**
2883fd0b676Sandi     * Callback for footnote end syntax
2893fd0b676Sandi     *
2903fd0b676Sandi     * All rendered content is moved to the $footnotes array and the old
2913fd0b676Sandi     * content is restored from $store again
2923fd0b676Sandi     *
2933fd0b676Sandi     * @author Andreas Gohr
2943fd0b676Sandi     */
2950cecf9d5Sandi    function footnote_close() {
29616ec3e37SAndreas Gohr        /** @var $fnid int takes track of seen footnotes, assures they are unique even across multiple docs FS#2841 */
29716ec3e37SAndreas Gohr        static $fnid = 0;
29816ec3e37SAndreas Gohr        // assign new footnote id (we start at 1)
29916ec3e37SAndreas Gohr        $fnid++;
3007764a90aSandi
301d74aace9Schris        // recover footnote into the stack and restore old content
302d74aace9Schris        $footnote = $this->doc;
3037764a90aSandi        $this->doc = $this->store;
3047764a90aSandi        $this->store = '';
305d74aace9Schris
306d74aace9Schris        // check to see if this footnote has been seen before
307d74aace9Schris        $i = array_search($footnote, $this->footnotes);
308d74aace9Schris
309d74aace9Schris        if ($i === false) {
310d74aace9Schris            // its a new footnote, add it to the $footnotes array
31116ec3e37SAndreas Gohr            $this->footnotes[$fnid] = $footnote;
312d74aace9Schris        } else {
31316ec3e37SAndreas Gohr            // seen this one before, save a placeholder
31416ec3e37SAndreas Gohr            $this->footnotes[$fnid] = "@@FNT".($i);
315d74aace9Schris        }
316d74aace9Schris
3176b379cbfSAndreas Gohr        // output the footnote reference and link
31816ec3e37SAndreas Gohr        $this->doc .= '<sup><a href="#fn__'.$fnid.'" id="fnt__'.$fnid.'" class="fn_top">'.$fnid.')</a></sup>';
3190cecf9d5Sandi    }
3200cecf9d5Sandi
3210cecf9d5Sandi    function listu_open() {
322a2d649c4Sandi        $this->doc .= '<ul>'.DOKU_LF;
3230cecf9d5Sandi    }
3240cecf9d5Sandi
3250cecf9d5Sandi    function listu_close() {
326a2d649c4Sandi        $this->doc .= '</ul>'.DOKU_LF;
3270cecf9d5Sandi    }
3280cecf9d5Sandi
3290cecf9d5Sandi    function listo_open() {
330a2d649c4Sandi        $this->doc .= '<ol>'.DOKU_LF;
3310cecf9d5Sandi    }
3320cecf9d5Sandi
3330cecf9d5Sandi    function listo_close() {
334a2d649c4Sandi        $this->doc .= '</ol>'.DOKU_LF;
3350cecf9d5Sandi    }
3360cecf9d5Sandi
3370cecf9d5Sandi    function listitem_open($level) {
33859869a4bSAnika Henke        $this->doc .= '<li class="level'.$level.'">';
3390cecf9d5Sandi    }
3400cecf9d5Sandi
3410cecf9d5Sandi    function listitem_close() {
342a2d649c4Sandi        $this->doc .= '</li>'.DOKU_LF;
3430cecf9d5Sandi    }
3440cecf9d5Sandi
3450cecf9d5Sandi    function listcontent_open() {
34690db23d7Schris        $this->doc .= '<div class="li">';
3470cecf9d5Sandi    }
3480cecf9d5Sandi
3490cecf9d5Sandi    function listcontent_close() {
35059869a4bSAnika Henke        $this->doc .= '</div>'.DOKU_LF;
3510cecf9d5Sandi    }
3520cecf9d5Sandi
3530cecf9d5Sandi    function unformatted($text) {
354a2d649c4Sandi        $this->doc .= $this->_xmlEntities($text);
3550cecf9d5Sandi    }
3560cecf9d5Sandi
3570cecf9d5Sandi    /**
3583fd0b676Sandi     * Execute PHP code if allowed
3593fd0b676Sandi     *
360d9764001SMichael Hamann     * @param  string   $text      PHP code that is either executed or printed
3615d568b99SChris Smith     * @param  string   $wrapper   html element to wrap result if $conf['phpok'] is okff
3625d568b99SChris Smith     *
3633fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3640cecf9d5Sandi     */
3655d568b99SChris Smith    function php($text, $wrapper='code') {
36635a56260SChris Smith        global $conf;
36735a56260SChris Smith
368d86d5af0SChris Smith        if($conf['phpok']){
369bad0b545Sandi            ob_start();
3704de671bcSandi            eval($text);
3713fd0b676Sandi            $this->doc .= ob_get_contents();
372bad0b545Sandi            ob_end_clean();
373d86d5af0SChris Smith        } else {
3745d568b99SChris Smith            $this->doc .= p_xhtml_cached_geshi($text, 'php', $wrapper);
375d86d5af0SChris Smith        }
3760cecf9d5Sandi    }
3770cecf9d5Sandi
37807f89c3cSAnika Henke    function phpblock($text) {
3795d568b99SChris Smith        $this->php($text, 'pre');
38007f89c3cSAnika Henke    }
38107f89c3cSAnika Henke
3820cecf9d5Sandi    /**
3833fd0b676Sandi     * Insert HTML if allowed
3843fd0b676Sandi     *
385d9764001SMichael Hamann     * @param  string   $text      html text
3865d568b99SChris Smith     * @param  string   $wrapper   html element to wrap result if $conf['htmlok'] is okff
3875d568b99SChris Smith     *
3883fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
3890cecf9d5Sandi     */
3905d568b99SChris Smith    function html($text, $wrapper='code') {
39135a56260SChris Smith        global $conf;
39235a56260SChris Smith
393d86d5af0SChris Smith        if($conf['htmlok']){
394a2d649c4Sandi            $this->doc .= $text;
395d86d5af0SChris Smith        } else {
3965d568b99SChris Smith            $this->doc .= p_xhtml_cached_geshi($text, 'html4strict', $wrapper);
397d86d5af0SChris Smith        }
3984de671bcSandi    }
3990cecf9d5Sandi
40007f89c3cSAnika Henke    function htmlblock($text) {
4015d568b99SChris Smith        $this->html($text, 'pre');
40207f89c3cSAnika Henke    }
40307f89c3cSAnika Henke
4040cecf9d5Sandi    function quote_open() {
40596331712SAnika Henke        $this->doc .= '<blockquote><div class="no">'.DOKU_LF;
4060cecf9d5Sandi    }
4070cecf9d5Sandi
4080cecf9d5Sandi    function quote_close() {
40996331712SAnika Henke        $this->doc .= '</div></blockquote>'.DOKU_LF;
4100cecf9d5Sandi    }
4110cecf9d5Sandi
4123d491f75SAndreas Gohr    function preformatted($text) {
413c9250713SAnika Henke        $this->doc .= '<pre class="code">' . trim($this->_xmlEntities($text),"\n\r") . '</pre>'. DOKU_LF;
4143d491f75SAndreas Gohr    }
4153d491f75SAndreas Gohr
4163d491f75SAndreas Gohr    function file($text, $language=null, $filename=null) {
4173d491f75SAndreas Gohr        $this->_highlight('file',$text,$language,$filename);
4183d491f75SAndreas Gohr    }
4193d491f75SAndreas Gohr
4203d491f75SAndreas Gohr    function code($text, $language=null, $filename=null) {
4213d491f75SAndreas Gohr        $this->_highlight('code',$text,$language,$filename);
4223d491f75SAndreas Gohr    }
4233d491f75SAndreas Gohr
4240cecf9d5Sandi    /**
4253d491f75SAndreas Gohr     * Use GeSHi to highlight language syntax in code and file blocks
4263fd0b676Sandi     *
4273fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
4280cecf9d5Sandi     */
4293d491f75SAndreas Gohr    function _highlight($type, $text, $language=null, $filename=null) {
4304de671bcSandi        global $conf;
4313d491f75SAndreas Gohr        global $ID;
4323d491f75SAndreas Gohr        global $lang;
4333d491f75SAndreas Gohr
4343d491f75SAndreas Gohr        if($filename){
435190c56e8SAndreas Gohr            // add icon
43627bf7924STom N Harris            list($ext) = mimetype($filename,false);
437190c56e8SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
438190c56e8SAndreas Gohr            $class = 'mediafile mf_'.$class;
439190c56e8SAndreas Gohr
4403d491f75SAndreas Gohr            $this->doc .= '<dl class="'.$type.'">'.DOKU_LF;
441190c56e8SAndreas Gohr            $this->doc .= '<dt><a href="'.exportlink($ID,'code',array('codeblock'=>$this->_codeblock)).'" title="'.$lang['download'].'" class="'.$class.'">';
4423d491f75SAndreas Gohr            $this->doc .= hsc($filename);
4433d491f75SAndreas Gohr            $this->doc .= '</a></dt>'.DOKU_LF.'<dd>';
4443d491f75SAndreas Gohr        }
4450cecf9d5Sandi
446d43aac1cSGina Haeussge        if ($text{0} == "\n") {
447d43aac1cSGina Haeussge            $text = substr($text, 1);
448d43aac1cSGina Haeussge        }
449d43aac1cSGina Haeussge        if (substr($text, -1) == "\n") {
450d43aac1cSGina Haeussge            $text = substr($text, 0, -1);
451d43aac1cSGina Haeussge        }
452d43aac1cSGina Haeussge
4530cecf9d5Sandi        if ( is_null($language) ) {
4543d491f75SAndreas Gohr            $this->doc .= '<pre class="'.$type.'">'.$this->_xmlEntities($text).'</pre>'.DOKU_LF;
4550cecf9d5Sandi        } else {
4563d491f75SAndreas Gohr            $class = 'code'; //we always need the code class to make the syntax highlighting apply
4573d491f75SAndreas Gohr            if($type != 'code') $class .= ' '.$type;
4583d491f75SAndreas Gohr
4593d491f75SAndreas Gohr            $this->doc .= "<pre class=\"$class $language\">".p_xhtml_cached_geshi($text, $language, '').'</pre>'.DOKU_LF;
4600cecf9d5Sandi        }
4613d491f75SAndreas Gohr
4623d491f75SAndreas Gohr        if($filename){
4633d491f75SAndreas Gohr            $this->doc .= '</dd></dl>'.DOKU_LF;
4643d491f75SAndreas Gohr        }
4653d491f75SAndreas Gohr
4663d491f75SAndreas Gohr        $this->_codeblock++;
4670cecf9d5Sandi    }
4680cecf9d5Sandi
4690cecf9d5Sandi    function acronym($acronym) {
4700cecf9d5Sandi
4710cecf9d5Sandi        if ( array_key_exists($acronym, $this->acronyms) ) {
4720cecf9d5Sandi
473433bef32Sandi            $title = $this->_xmlEntities($this->acronyms[$acronym]);
4740cecf9d5Sandi
475940db3a3SAnika Henke            $this->doc .= '<abbr title="'.$title
476940db3a3SAnika Henke                .'">'.$this->_xmlEntities($acronym).'</abbr>';
4770cecf9d5Sandi
4780cecf9d5Sandi        } else {
479a2d649c4Sandi            $this->doc .= $this->_xmlEntities($acronym);
4800cecf9d5Sandi        }
4810cecf9d5Sandi    }
4820cecf9d5Sandi
4830cecf9d5Sandi    function smiley($smiley) {
4840cecf9d5Sandi        if ( array_key_exists($smiley, $this->smileys) ) {
485433bef32Sandi            $title = $this->_xmlEntities($this->smileys[$smiley]);
486f62ea8a1Sandi            $this->doc .= '<img src="'.DOKU_BASE.'lib/images/smileys/'.$this->smileys[$smiley].
4878e38227fSAnika Henke                '" class="icon" alt="'.
488433bef32Sandi                    $this->_xmlEntities($smiley).'" />';
4890cecf9d5Sandi        } else {
490a2d649c4Sandi            $this->doc .= $this->_xmlEntities($smiley);
4910cecf9d5Sandi        }
4920cecf9d5Sandi    }
4930cecf9d5Sandi
494f62ea8a1Sandi    /*
4954de671bcSandi    * not used
4960cecf9d5Sandi    function wordblock($word) {
4970cecf9d5Sandi        if ( array_key_exists($word, $this->badwords) ) {
498a2d649c4Sandi            $this->doc .= '** BLEEP **';
4990cecf9d5Sandi        } else {
500a2d649c4Sandi            $this->doc .= $this->_xmlEntities($word);
5010cecf9d5Sandi        }
5020cecf9d5Sandi    }
5034de671bcSandi    */
5040cecf9d5Sandi
5050cecf9d5Sandi    function entity($entity) {
5060cecf9d5Sandi        if ( array_key_exists($entity, $this->entities) ) {
507a2d649c4Sandi            $this->doc .= $this->entities[$entity];
5080cecf9d5Sandi        } else {
509a2d649c4Sandi            $this->doc .= $this->_xmlEntities($entity);
5100cecf9d5Sandi        }
5110cecf9d5Sandi    }
5120cecf9d5Sandi
5130cecf9d5Sandi    function multiplyentity($x, $y) {
514a2d649c4Sandi        $this->doc .= "$x&times;$y";
5150cecf9d5Sandi    }
5160cecf9d5Sandi
5170cecf9d5Sandi    function singlequoteopening() {
51871b40da2SAnika Henke        global $lang;
51971b40da2SAnika Henke        $this->doc .= $lang['singlequoteopening'];
5200cecf9d5Sandi    }
5210cecf9d5Sandi
5220cecf9d5Sandi    function singlequoteclosing() {
52371b40da2SAnika Henke        global $lang;
52471b40da2SAnika Henke        $this->doc .= $lang['singlequoteclosing'];
5250cecf9d5Sandi    }
5260cecf9d5Sandi
52757d757d1SAndreas Gohr    function apostrophe() {
52857d757d1SAndreas Gohr        global $lang;
529a8bd192aSAndreas Gohr        $this->doc .= $lang['apostrophe'];
53057d757d1SAndreas Gohr    }
53157d757d1SAndreas Gohr
5320cecf9d5Sandi    function doublequoteopening() {
53371b40da2SAnika Henke        global $lang;
53471b40da2SAnika Henke        $this->doc .= $lang['doublequoteopening'];
5350cecf9d5Sandi    }
5360cecf9d5Sandi
5370cecf9d5Sandi    function doublequoteclosing() {
53871b40da2SAnika Henke        global $lang;
53971b40da2SAnika Henke        $this->doc .= $lang['doublequoteclosing'];
5400cecf9d5Sandi    }
5410cecf9d5Sandi
5420cecf9d5Sandi    /**
5430cecf9d5Sandi     */
5440cecf9d5Sandi    function camelcaselink($link) {
54511d0aa47Sandi        $this->internallink($link,$link);
5460cecf9d5Sandi    }
5470cecf9d5Sandi
5480b7c14c2Sandi
5490ea51e63SMatt Perry    function locallink($hash, $name = null){
5500b7c14c2Sandi        global $ID;
5510b7c14c2Sandi        $name  = $this->_getLinkTitle($name, $hash, $isImage);
5520b7c14c2Sandi        $hash  = $this->_headerToLink($hash);
553e260f93bSAnika Henke        $title = $ID.' ↵';
5540b7c14c2Sandi        $this->doc .= '<a href="#'.$hash.'" title="'.$title.'" class="wikilink1">';
5550b7c14c2Sandi        $this->doc .= $name;
5560b7c14c2Sandi        $this->doc .= '</a>';
5570b7c14c2Sandi    }
5580b7c14c2Sandi
559cffcc403Sandi    /**
5603fd0b676Sandi     * Render an internal Wiki Link
5613fd0b676Sandi     *
562fe9ec250SChris Smith     * $search,$returnonly & $linktype are not for the renderer but are used
563cffcc403Sandi     * elsewhere - no need to implement them in other renderers
5643fd0b676Sandi     *
5653fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
566cffcc403Sandi     */
5670ea51e63SMatt Perry    function internallink($id, $name = null, $search=null,$returnonly=false,$linktype='content') {
568ba11bd29Sandi        global $conf;
56937e34a5eSandi        global $ID;
570c4dda6afSAnika Henke        global $INFO;
57144653a53SAdrian Lang
5723d5e07d9SAdrian Lang        $params = '';
5733d5e07d9SAdrian Lang        $parts = explode('?', $id, 2);
5743d5e07d9SAdrian Lang        if (count($parts) === 2) {
5753d5e07d9SAdrian Lang            $id = $parts[0];
5763d5e07d9SAdrian Lang            $params = $parts[1];
57744653a53SAdrian Lang        }
57844653a53SAdrian Lang
579fda14ffcSIzidor Matušov        // For empty $id we need to know the current $ID
580fda14ffcSIzidor Matušov        // We need this check because _simpleTitle needs
581fda14ffcSIzidor Matušov        // correct $id and resolve_pageid() use cleanID($id)
582fda14ffcSIzidor Matušov        // (some things could be lost)
583fda14ffcSIzidor Matušov        if ($id === '') {
584fda14ffcSIzidor Matušov            $id = $ID;
585fda14ffcSIzidor Matušov        }
586fda14ffcSIzidor Matušov
5870339c872Sjan        // default name is based on $id as given
5880339c872Sjan        $default = $this->_simpleTitle($id);
589ad32e47eSAndreas Gohr
5900339c872Sjan        // now first resolve and clean up the $id
59137e34a5eSandi        resolve_pageid(getNS($ID),$id,$exists);
592fda14ffcSIzidor Matušov
593fe9ec250SChris Smith        $name = $this->_getLinkTitle($name, $default, $isImage, $id, $linktype);
5940e1c636eSandi        if ( !$isImage ) {
5950e1c636eSandi            if ( $exists ) {
596ba11bd29Sandi                $class='wikilink1';
5970cecf9d5Sandi            } else {
598ba11bd29Sandi                $class='wikilink2';
59944a6b4c7SAndreas Gohr                $link['rel']='nofollow';
6000cecf9d5Sandi            }
6010cecf9d5Sandi        } else {
602ba11bd29Sandi            $class='media';
6030cecf9d5Sandi        }
6040cecf9d5Sandi
605a1685bedSandi        //keep hash anchor
606ce6b63d9Schris        list($id,$hash) = explode('#',$id,2);
607943dedc6SAndreas Gohr        if(!empty($hash)) $hash = $this->_headerToLink($hash);
608a1685bedSandi
609ba11bd29Sandi        //prepare for formating
610ba11bd29Sandi        $link['target'] = $conf['target']['wiki'];
611ba11bd29Sandi        $link['style']  = '';
612ba11bd29Sandi        $link['pre']    = '';
613ba11bd29Sandi        $link['suf']    = '';
61440eb54bbSjan        // highlight link to current page
615c4dda6afSAnika Henke        if ($id == $INFO['id']) {
61692795d04Sandi            $link['pre']    = '<span class="curid">';
61792795d04Sandi            $link['suf']    = '</span>';
61840eb54bbSjan        }
6195e163278SAndreas Gohr        $link['more']   = '';
620ba11bd29Sandi        $link['class']  = $class;
62144653a53SAdrian Lang        $link['url']    = wl($id, $params);
622ba11bd29Sandi        $link['name']   = $name;
623ba11bd29Sandi        $link['title']  = $id;
624723d78dbSandi        //add search string
625723d78dbSandi        if($search){
626546d3a99SAndreas Gohr            ($conf['userewrite']) ? $link['url'].='?' : $link['url'].='&amp;';
627546d3a99SAndreas Gohr            if(is_array($search)){
628546d3a99SAndreas Gohr                $search = array_map('rawurlencode',$search);
629546d3a99SAndreas Gohr                $link['url'] .= 's[]='.join('&amp;s[]=',$search);
630546d3a99SAndreas Gohr            }else{
631546d3a99SAndreas Gohr                $link['url'] .= 's='.rawurlencode($search);
632546d3a99SAndreas Gohr            }
633723d78dbSandi        }
634723d78dbSandi
635a1685bedSandi        //keep hash
636a1685bedSandi        if($hash) $link['url'].='#'.$hash;
637a1685bedSandi
638ba11bd29Sandi        //output formatted
639cffcc403Sandi        if($returnonly){
640cffcc403Sandi            return $this->_formatLink($link);
641cffcc403Sandi        }else{
642a2d649c4Sandi            $this->doc .= $this->_formatLink($link);
6430cecf9d5Sandi        }
644cffcc403Sandi    }
6450cecf9d5Sandi
6460ea51e63SMatt Perry    function externallink($url, $name = null) {
647b625487dSandi        global $conf;
6480cecf9d5Sandi
649433bef32Sandi        $name = $this->_getLinkTitle($name, $url, $isImage);
6506f0c5dbfSandi
651b52b1596SAndreas Gohr        // url might be an attack vector, only allow registered protocols
652b52b1596SAndreas Gohr        if(is_null($this->schemes)) $this->schemes = getSchemes();
653b52b1596SAndreas Gohr        list($scheme) = explode('://',$url);
654b52b1596SAndreas Gohr        $scheme = strtolower($scheme);
655b52b1596SAndreas Gohr        if(!in_array($scheme,$this->schemes)) $url = '';
656b52b1596SAndreas Gohr
657b52b1596SAndreas Gohr        // is there still an URL?
658b52b1596SAndreas Gohr        if(!$url){
659b52b1596SAndreas Gohr            $this->doc .= $name;
660b52b1596SAndreas Gohr            return;
661b52b1596SAndreas Gohr        }
662b52b1596SAndreas Gohr
663b52b1596SAndreas Gohr        // set class
6640cecf9d5Sandi        if ( !$isImage ) {
665b625487dSandi            $class='urlextern';
6660cecf9d5Sandi        } else {
667b625487dSandi            $class='media';
6680cecf9d5Sandi        }
6690cecf9d5Sandi
670b625487dSandi        //prepare for formating
671b625487dSandi        $link['target'] = $conf['target']['extern'];
672b625487dSandi        $link['style']  = '';
673b625487dSandi        $link['pre']    = '';
674b625487dSandi        $link['suf']    = '';
6755e163278SAndreas Gohr        $link['more']   = '';
676b625487dSandi        $link['class']  = $class;
677b625487dSandi        $link['url']    = $url;
678e1c10e4dSchris
679b625487dSandi        $link['name']   = $name;
680433bef32Sandi        $link['title']  = $this->_xmlEntities($url);
681b625487dSandi        if($conf['relnofollow']) $link['more'] .= ' rel="nofollow"';
6820cecf9d5Sandi
683b625487dSandi        //output formatted
684a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6850cecf9d5Sandi    }
6860cecf9d5Sandi
6870cecf9d5Sandi    /**
6880cecf9d5Sandi    */
6890ea51e63SMatt Perry    function interwikilink($match, $name = null, $wikiName, $wikiUri) {
690b625487dSandi        global $conf;
6910cecf9d5Sandi
69297a3e4e3Sandi        $link = array();
69397a3e4e3Sandi        $link['target'] = $conf['target']['interwiki'];
69497a3e4e3Sandi        $link['pre']    = '';
69597a3e4e3Sandi        $link['suf']    = '';
6965e163278SAndreas Gohr        $link['more']   = '';
697433bef32Sandi        $link['name']   = $this->_getLinkTitle($name, $wikiUri, $isImage);
6980cecf9d5Sandi
69997a3e4e3Sandi        //get interwiki URL
7001f82fabeSAndreas Gohr        $url = $this->_resolveInterWiki($wikiName,$wikiUri);
7010cecf9d5Sandi
70297a3e4e3Sandi        if ( !$isImage ) {
7039d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$wikiName);
7049d2ddea4SAndreas Gohr            $link['class'] = "interwiki iw_$class";
7051c2d1019SAndreas Gohr        } else {
7061c2d1019SAndreas Gohr            $link['class'] = 'media';
70797a3e4e3Sandi        }
7080cecf9d5Sandi
70997a3e4e3Sandi        //do we stay at the same server? Use local target
71097a3e4e3Sandi        if( strpos($url,DOKU_URL) === 0 ){
71197a3e4e3Sandi            $link['target'] = $conf['target']['wiki'];
71297a3e4e3Sandi        }
7130cecf9d5Sandi
71497a3e4e3Sandi        $link['url'] = $url;
71597a3e4e3Sandi        $link['title'] = htmlspecialchars($link['url']);
71697a3e4e3Sandi
71797a3e4e3Sandi        //output formatted
718a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7190cecf9d5Sandi    }
7200cecf9d5Sandi
7210cecf9d5Sandi    /**
7220cecf9d5Sandi     */
7230ea51e63SMatt Perry    function windowssharelink($url, $name = null) {
7241d47afe1Sandi        global $conf;
7251d47afe1Sandi        global $lang;
7261d47afe1Sandi        //simple setup
7271d47afe1Sandi        $link['target'] = $conf['target']['windows'];
7281d47afe1Sandi        $link['pre']    = '';
7291d47afe1Sandi        $link['suf']   = '';
7301d47afe1Sandi        $link['style']  = '';
7310cecf9d5Sandi
732433bef32Sandi        $link['name'] = $this->_getLinkTitle($name, $url, $isImage);
7330cecf9d5Sandi        if ( !$isImage ) {
7341d47afe1Sandi            $link['class'] = 'windows';
7350cecf9d5Sandi        } else {
7361d47afe1Sandi            $link['class'] = 'media';
7370cecf9d5Sandi        }
7380cecf9d5Sandi
739433bef32Sandi        $link['title'] = $this->_xmlEntities($url);
7401d47afe1Sandi        $url = str_replace('\\','/',$url);
7411d47afe1Sandi        $url = 'file:///'.$url;
7421d47afe1Sandi        $link['url'] = $url;
7430cecf9d5Sandi
7441d47afe1Sandi        //output formatted
745a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7460cecf9d5Sandi    }
7470cecf9d5Sandi
7480ea51e63SMatt Perry    function emaillink($address, $name = null) {
74971352defSandi        global $conf;
75071352defSandi        //simple setup
75171352defSandi        $link = array();
75271352defSandi        $link['target'] = '';
75371352defSandi        $link['pre']    = '';
75471352defSandi        $link['suf']   = '';
75571352defSandi        $link['style']  = '';
75671352defSandi        $link['more']   = '';
7570cecf9d5Sandi
758c078fc55SAndreas Gohr        $name = $this->_getLinkTitle($name, '', $isImage);
7590cecf9d5Sandi        if ( !$isImage ) {
760be96545cSAnika Henke            $link['class']='mail';
7610cecf9d5Sandi        } else {
762be96545cSAnika Henke            $link['class']='media';
7630cecf9d5Sandi        }
7640cecf9d5Sandi
76507738714SAndreas Gohr        $address = $this->_xmlEntities($address);
76600a7b5adSEsther Brunner        $address = obfuscate($address);
76700a7b5adSEsther Brunner        $title   = $address;
7688c128049SAndreas Gohr
76971352defSandi        if(empty($name)){
77000a7b5adSEsther Brunner            $name = $address;
77171352defSandi        }
7720cecf9d5Sandi
773776b36ecSAndreas Gohr        if($conf['mailguard'] == 'visible') $address = rawurlencode($address);
774776b36ecSAndreas Gohr
775776b36ecSAndreas Gohr        $link['url']   = 'mailto:'.$address;
77671352defSandi        $link['name']  = $name;
77771352defSandi        $link['title'] = $title;
7780cecf9d5Sandi
77971352defSandi        //output formatted
780a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7810cecf9d5Sandi    }
7820cecf9d5Sandi
7830ea51e63SMatt Perry    function internalmedia ($src, $title=null, $align=null, $width=null,
7840ea51e63SMatt Perry                            $height=null, $cache=null, $linking=null) {
78537e34a5eSandi        global $ID;
78691df343aSAndreas Gohr        list($src,$hash) = explode('#',$src,2);
78737e34a5eSandi        resolve_mediaid(getNS($ID),$src, $exists);
7880cecf9d5Sandi
789d98d4540SBen Coburn        $noLink = false;
7908acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
791b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
7923685f775Sandi
79327bf7924STom N Harris        list($ext,$mime,$dl) = mimetype($src,false);
794b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
795dc673a5bSjoe.lapp            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),($linking=='direct'));
7961c882ba8SAndreas Gohr        }elseif($mime == 'application/x-shockwave-flash' && $render){
7972ca14335SEsther Brunner            // don't link flash movies
79844881bd0Shenning.noren            $noLink = true;
79955efc227SAndreas Gohr        }else{
8002ca14335SEsther Brunner            // add file icons
8019d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
8029d2ddea4SAndreas Gohr            $link['class'] .= ' mediafile mf_'.$class;
8036de3759aSAndreas Gohr            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),true);
80491328684SMichael Hamann            if ($exists) $link['title'] .= ' (' . filesize_h(filesize(mediaFN($src))).')';
80555efc227SAndreas Gohr        }
8063685f775Sandi
80791df343aSAndreas Gohr        if($hash) $link['url'] .= '#'.$hash;
80891df343aSAndreas Gohr
8096fe20453SGina Haeussge        //markup non existing files
8104a24b459SKate Arzamastseva        if (!$exists) {
8116fe20453SGina Haeussge            $link['class'] .= ' wikilink2';
8124a24b459SKate Arzamastseva        }
8136fe20453SGina Haeussge
8143685f775Sandi        //output formatted
815dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
8162ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
8170cecf9d5Sandi    }
8180cecf9d5Sandi
8190ea51e63SMatt Perry    function externalmedia ($src, $title=null, $align=null, $width=null,
8200ea51e63SMatt Perry                            $height=null, $cache=null, $linking=null) {
82191df343aSAndreas Gohr        list($src,$hash) = explode('#',$src,2);
822d98d4540SBen Coburn        $noLink = false;
8238acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
824b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
825b739ff0fSPierre Spring
826b739ff0fSPierre Spring        $link['url']    = ml($src,array('cache'=>$cache));
8273685f775Sandi
82827bf7924STom N Harris        list($ext,$mime,$dl) = mimetype($src,false);
829b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
8302ca14335SEsther Brunner            // link only jpeg images
83144881bd0Shenning.noren            // if ($ext != 'jpg' && $ext != 'jpeg') $noLink = true;
8321c882ba8SAndreas Gohr        }elseif($mime == 'application/x-shockwave-flash' && $render){
8332ca14335SEsther Brunner            // don't link flash movies
83444881bd0Shenning.noren            $noLink = true;
8352ca14335SEsther Brunner        }else{
8362ca14335SEsther Brunner            // add file icons
83727bf7924STom N Harris            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
83827bf7924STom N Harris            $link['class'] .= ' mediafile mf_'.$class;
8392ca14335SEsther Brunner        }
8402ca14335SEsther Brunner
84191df343aSAndreas Gohr        if($hash) $link['url'] .= '#'.$hash;
84291df343aSAndreas Gohr
8433685f775Sandi        //output formatted
844dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
8452ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
8460cecf9d5Sandi    }
8470cecf9d5Sandi
8484826ab45Sandi    /**
8493db95becSAndreas Gohr     * Renders an RSS feed
850b625487dSandi     *
851b625487dSandi     * @author Andreas Gohr <andi@splitbrain.org>
852b625487dSandi     */
8533db95becSAndreas Gohr    function rss ($url,$params){
854b625487dSandi        global $lang;
8553db95becSAndreas Gohr        global $conf;
8563db95becSAndreas Gohr
8573db95becSAndreas Gohr        require_once(DOKU_INC.'inc/FeedParser.php');
8583db95becSAndreas Gohr        $feed = new FeedParser();
85900077af8SAndreas Gohr        $feed->set_feed_url($url);
860b625487dSandi
861b625487dSandi        //disable warning while fetching
862bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { $elvl = error_reporting(E_ERROR); }
8633db95becSAndreas Gohr        $rc = $feed->init();
864bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { error_reporting($elvl); }
865b625487dSandi
8663db95becSAndreas Gohr        //decide on start and end
8673db95becSAndreas Gohr        if($params['reverse']){
8683db95becSAndreas Gohr            $mod = -1;
8693db95becSAndreas Gohr            $start = $feed->get_item_quantity()-1;
8703db95becSAndreas Gohr            $end   = $start - ($params['max']);
871b2a412b0SAndreas Gohr            $end   = ($end < -1) ? -1 : $end;
8723db95becSAndreas Gohr        }else{
8733db95becSAndreas Gohr            $mod   = 1;
8743db95becSAndreas Gohr            $start = 0;
8753db95becSAndreas Gohr            $end   = $feed->get_item_quantity();
876d91ab76fSMatt Perry            $end   = ($end > $params['max']) ? $params['max'] : $end;
8773db95becSAndreas Gohr        }
8783db95becSAndreas Gohr
879a2d649c4Sandi        $this->doc .= '<ul class="rss">';
8803db95becSAndreas Gohr        if($rc){
8813db95becSAndreas Gohr            for ($x = $start; $x != $end; $x += $mod) {
8821bde1582SAndreas Gohr                $item = $feed->get_item($x);
8833db95becSAndreas Gohr                $this->doc .= '<li><div class="li">';
884d2ea3363SAndreas Gohr                // support feeds without links
885d2ea3363SAndreas Gohr                $lnkurl = $item->get_permalink();
886d2ea3363SAndreas Gohr                if($lnkurl){
887793361f8SAndreas Gohr                    // title is escaped by SimplePie, we unescape here because it
888793361f8SAndreas Gohr                    // is escaped again in externallink() FS#1705
8891bde1582SAndreas Gohr                    $this->externallink($item->get_permalink(),
890b9b9b28bSMichael Hamann                                        html_entity_decode($item->get_title(), ENT_QUOTES, 'UTF-8'));
891d2ea3363SAndreas Gohr                }else{
892d2ea3363SAndreas Gohr                    $this->doc .= ' '.$item->get_title();
893d2ea3363SAndreas Gohr                }
8943db95becSAndreas Gohr                if($params['author']){
8951bde1582SAndreas Gohr                    $author = $item->get_author(0);
8961bde1582SAndreas Gohr                    if($author){
8971bde1582SAndreas Gohr                        $name = $author->get_name();
8981bde1582SAndreas Gohr                        if(!$name) $name = $author->get_email();
8991bde1582SAndreas Gohr                        if($name) $this->doc .= ' '.$lang['by'].' '.$name;
9001bde1582SAndreas Gohr                    }
9013db95becSAndreas Gohr                }
9023db95becSAndreas Gohr                if($params['date']){
9032e7e0c29SAndreas Gohr                    $this->doc .= ' ('.$item->get_local_date($conf['dformat']).')';
9043db95becSAndreas Gohr                }
9051bde1582SAndreas Gohr                if($params['details']){
9063db95becSAndreas Gohr                    $this->doc .= '<div class="detail">';
907173dccb7STom N Harris                    if($conf['htmlok']){
9081bde1582SAndreas Gohr                        $this->doc .= $item->get_description();
9093db95becSAndreas Gohr                    }else{
9101bde1582SAndreas Gohr                        $this->doc .= strip_tags($item->get_description());
9113db95becSAndreas Gohr                    }
9123db95becSAndreas Gohr                    $this->doc .= '</div>';
9133db95becSAndreas Gohr                }
9143db95becSAndreas Gohr
9153db95becSAndreas Gohr                $this->doc .= '</div></li>';
916b625487dSandi            }
917b625487dSandi        }else{
9183db95becSAndreas Gohr            $this->doc .= '<li><div class="li">';
919a2d649c4Sandi            $this->doc .= '<em>'.$lang['rssfailed'].'</em>';
920b625487dSandi            $this->externallink($url);
92145e147ccSAndreas Gohr            if($conf['allowdebug']){
92245e147ccSAndreas Gohr                $this->doc .= '<!--'.hsc($feed->error).'-->';
92345e147ccSAndreas Gohr            }
9243db95becSAndreas Gohr            $this->doc .= '</div></li>';
925b625487dSandi        }
926a2d649c4Sandi        $this->doc .= '</ul>';
927b625487dSandi    }
928b625487dSandi
9290cecf9d5Sandi    // $numrows not yet implemented
930619736fdSAdrian Lang    function table_open($maxcols = null, $numrows = null, $pos = null){
93190df9a4dSAdrian Lang        global $lang;
932b5742cedSPierre Spring        // initialize the row counter used for classes
933b5742cedSPierre Spring        $this->_counter['row_counter'] = 0;
934619736fdSAdrian Lang        $class = 'table';
935619736fdSAdrian Lang        if ($pos !== null) {
936619736fdSAdrian Lang            $class .= ' ' . $this->startSectionEdit($pos, 'table');
937619736fdSAdrian Lang        }
938619736fdSAdrian Lang        $this->doc .= '<div class="' . $class . '"><table class="inline">' .
939619736fdSAdrian Lang                      DOKU_LF;
9400cecf9d5Sandi    }
9410cecf9d5Sandi
942619736fdSAdrian Lang    function table_close($pos = null){
943a8574918SAnika Henke        $this->doc .= '</table></div>'.DOKU_LF;
944619736fdSAdrian Lang        if ($pos !== null) {
94590df9a4dSAdrian Lang            $this->finishSectionEdit($pos);
9460cecf9d5Sandi        }
947619736fdSAdrian Lang    }
9480cecf9d5Sandi
949*f05a1cc5SGerrit Uitslag    function tablethead_open(){
950*f05a1cc5SGerrit Uitslag        $this->doc .= DOKU_TAB . '<thead>' . DOKU_LF;
951*f05a1cc5SGerrit Uitslag    }
952*f05a1cc5SGerrit Uitslag
953*f05a1cc5SGerrit Uitslag    function tablethead_close(){
954*f05a1cc5SGerrit Uitslag        $this->doc .= DOKU_TAB . '</thead>' . DOKU_LF;
955*f05a1cc5SGerrit Uitslag    }
956*f05a1cc5SGerrit Uitslag
9570cecf9d5Sandi    function tablerow_open(){
958b5742cedSPierre Spring        // initialize the cell counter used for classes
959b5742cedSPierre Spring        $this->_counter['cell_counter'] = 0;
960b5742cedSPierre Spring        $class = 'row' . $this->_counter['row_counter']++;
961b5742cedSPierre Spring        $this->doc .= DOKU_TAB . '<tr class="'.$class.'">' . DOKU_LF . DOKU_TAB . DOKU_TAB;
9620cecf9d5Sandi    }
9630cecf9d5Sandi
9640cecf9d5Sandi    function tablerow_close(){
965a2d649c4Sandi        $this->doc .= DOKU_LF . DOKU_TAB . '</tr>' . DOKU_LF;
9660cecf9d5Sandi    }
9670cecf9d5Sandi
9680ea51e63SMatt Perry    function tableheader_open($colspan = 1, $align = null, $rowspan = 1){
969b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
9700cecf9d5Sandi        if ( !is_null($align) ) {
971b5742cedSPierre Spring            $class .= ' '.$align.'align';
9720cecf9d5Sandi        }
973b5742cedSPierre Spring        $class .= '"';
974b5742cedSPierre Spring        $this->doc .= '<th ' . $class;
9750cecf9d5Sandi        if ( $colspan > 1 ) {
976a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
977a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
9780cecf9d5Sandi        }
97925b97867Shakan.sandell        if ( $rowspan > 1 ) {
98025b97867Shakan.sandell            $this->doc .= ' rowspan="'.$rowspan.'"';
98125b97867Shakan.sandell        }
982a2d649c4Sandi        $this->doc .= '>';
9830cecf9d5Sandi    }
9840cecf9d5Sandi
9850cecf9d5Sandi    function tableheader_close(){
986a2d649c4Sandi        $this->doc .= '</th>';
9870cecf9d5Sandi    }
9880cecf9d5Sandi
9890ea51e63SMatt Perry    function tablecell_open($colspan = 1, $align = null, $rowspan = 1){
990b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
9910cecf9d5Sandi        if ( !is_null($align) ) {
992b5742cedSPierre Spring            $class .= ' '.$align.'align';
9930cecf9d5Sandi        }
994b5742cedSPierre Spring        $class .= '"';
995b5742cedSPierre Spring        $this->doc .= '<td '.$class;
9960cecf9d5Sandi        if ( $colspan > 1 ) {
997a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
998a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
9990cecf9d5Sandi        }
100025b97867Shakan.sandell        if ( $rowspan > 1 ) {
100125b97867Shakan.sandell            $this->doc .= ' rowspan="'.$rowspan.'"';
100225b97867Shakan.sandell        }
1003a2d649c4Sandi        $this->doc .= '>';
10040cecf9d5Sandi    }
10050cecf9d5Sandi
10060cecf9d5Sandi    function tablecell_close(){
1007a2d649c4Sandi        $this->doc .= '</td>';
10080cecf9d5Sandi    }
10090cecf9d5Sandi
10100cecf9d5Sandi    //----------------------------------------------------------
10110cecf9d5Sandi    // Utils
10120cecf9d5Sandi
1013ba11bd29Sandi    /**
10143fd0b676Sandi     * Build a link
10153fd0b676Sandi     *
10163fd0b676Sandi     * Assembles all parts defined in $link returns HTML for the link
1017ba11bd29Sandi     *
1018ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
1019ba11bd29Sandi     */
1020433bef32Sandi    function _formatLink($link){
1021ba11bd29Sandi        //make sure the url is XHTML compliant (skip mailto)
1022ba11bd29Sandi        if(substr($link['url'],0,7) != 'mailto:'){
1023ba11bd29Sandi            $link['url'] = str_replace('&','&amp;',$link['url']);
1024ba11bd29Sandi            $link['url'] = str_replace('&amp;amp;','&amp;',$link['url']);
1025ba11bd29Sandi        }
1026ba11bd29Sandi        //remove double encodings in titles
1027ba11bd29Sandi        $link['title'] = str_replace('&amp;amp;','&amp;',$link['title']);
1028ba11bd29Sandi
1029453493f2SAndreas Gohr        // be sure there are no bad chars in url or title
1030453493f2SAndreas Gohr        // (we can't do this for name because it can contain an img tag)
1031453493f2SAndreas Gohr        $link['url']   = strtr($link['url'],array('>'=>'%3E','<'=>'%3C','"'=>'%22'));
1032453493f2SAndreas Gohr        $link['title'] = strtr($link['title'],array('>'=>'&gt;','<'=>'&lt;','"'=>'&quot;'));
1033453493f2SAndreas Gohr
1034ba11bd29Sandi        $ret  = '';
1035ba11bd29Sandi        $ret .= $link['pre'];
1036ba11bd29Sandi        $ret .= '<a href="'.$link['url'].'"';
1037bb4866bdSchris        if(!empty($link['class']))  $ret .= ' class="'.$link['class'].'"';
1038bb4866bdSchris        if(!empty($link['target'])) $ret .= ' target="'.$link['target'].'"';
1039bb4866bdSchris        if(!empty($link['title']))  $ret .= ' title="'.$link['title'].'"';
1040bb4866bdSchris        if(!empty($link['style']))  $ret .= ' style="'.$link['style'].'"';
104144a6b4c7SAndreas Gohr        if(!empty($link['rel']))    $ret .= ' rel="'.$link['rel'].'"';
1042bb4866bdSchris        if(!empty($link['more']))   $ret .= ' '.$link['more'];
1043ba11bd29Sandi        $ret .= '>';
1044ba11bd29Sandi        $ret .= $link['name'];
1045ba11bd29Sandi        $ret .= '</a>';
1046ba11bd29Sandi        $ret .= $link['suf'];
1047ba11bd29Sandi        return $ret;
1048ba11bd29Sandi    }
1049ba11bd29Sandi
1050ba11bd29Sandi    /**
10513fd0b676Sandi     * Renders internal and external media
10523fd0b676Sandi     *
10533fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
10543fd0b676Sandi     */
10550ea51e63SMatt Perry    function _media ($src, $title=null, $align=null, $width=null,
10560ea51e63SMatt Perry                      $height=null, $cache=null, $render = true) {
10573fd0b676Sandi
10583fd0b676Sandi        $ret = '';
10593fd0b676Sandi
1060ecebf3a8SAndreas Gohr        list($ext,$mime,$dl) = mimetype($src);
10613fd0b676Sandi        if(substr($mime,0,5) == 'image'){
1062b739ff0fSPierre Spring            // first get the $title
1063b739ff0fSPierre Spring            if (!is_null($title)) {
1064b739ff0fSPierre Spring                $title  = $this->_xmlEntities($title);
1065b739ff0fSPierre Spring            }elseif($ext == 'jpg' || $ext == 'jpeg'){
1066b739ff0fSPierre Spring                //try to use the caption from IPTC/EXIF
1067b739ff0fSPierre Spring                require_once(DOKU_INC.'inc/JpegMeta.php');
106867f9913dSAndreas Gohr                $jpeg =new JpegMeta(mediaFN($src));
1069b739ff0fSPierre Spring                if($jpeg !== false) $cap = $jpeg->getTitle();
1070b739ff0fSPierre Spring                if($cap){
1071b739ff0fSPierre Spring                    $title = $this->_xmlEntities($cap);
1072b739ff0fSPierre Spring                }
1073b739ff0fSPierre Spring            }
1074b739ff0fSPierre Spring            if (!$render) {
1075b739ff0fSPierre Spring                // if the picture is not supposed to be rendered
1076b739ff0fSPierre Spring                // return the title of the picture
1077b739ff0fSPierre Spring                if (!$title) {
1078b739ff0fSPierre Spring                    // just show the sourcename
10793009a773SAndreas Gohr                    $title = $this->_xmlEntities(utf8_basename(noNS($src)));
1080b739ff0fSPierre Spring                }
1081b739ff0fSPierre Spring                return $title;
1082b739ff0fSPierre Spring            }
10833fd0b676Sandi            //add image tag
10846de3759aSAndreas Gohr            $ret .= '<img src="'.ml($src,array('w'=>$width,'h'=>$height,'cache'=>$cache)).'"';
10853fd0b676Sandi            $ret .= ' class="media'.$align.'"';
10863fd0b676Sandi
1087b739ff0fSPierre Spring            if ($title) {
1088b739ff0fSPierre Spring                $ret .= ' title="' . $title . '"';
1089b739ff0fSPierre Spring                $ret .= ' alt="'   . $title .'"';
10903fd0b676Sandi            }else{
10913fd0b676Sandi                $ret .= ' alt=""';
10923fd0b676Sandi            }
10933fd0b676Sandi
10943fd0b676Sandi            if ( !is_null($width) )
10953fd0b676Sandi                $ret .= ' width="'.$this->_xmlEntities($width).'"';
10963fd0b676Sandi
10973fd0b676Sandi            if ( !is_null($height) )
10983fd0b676Sandi                $ret .= ' height="'.$this->_xmlEntities($height).'"';
10993fd0b676Sandi
11003fd0b676Sandi            $ret .= ' />';
11013fd0b676Sandi
11023fd0b676Sandi        }elseif($mime == 'application/x-shockwave-flash'){
11031c882ba8SAndreas Gohr            if (!$render) {
11041c882ba8SAndreas Gohr                // if the flash is not supposed to be rendered
11051c882ba8SAndreas Gohr                // return the title of the flash
11061c882ba8SAndreas Gohr                if (!$title) {
11071c882ba8SAndreas Gohr                    // just show the sourcename
11083009a773SAndreas Gohr                    $title = utf8_basename(noNS($src));
11091c882ba8SAndreas Gohr                }
111007bf32b2SAndreas Gohr                return $this->_xmlEntities($title);
11111c882ba8SAndreas Gohr            }
11121c882ba8SAndreas Gohr
111307bf32b2SAndreas Gohr            $att = array();
111407bf32b2SAndreas Gohr            $att['class'] = "media$align";
111507bf32b2SAndreas Gohr            if($align == 'right') $att['align'] = 'right';
111607bf32b2SAndreas Gohr            if($align == 'left')  $att['align'] = 'left';
1117c471e6a6SAndreas Gohr            $ret .= html_flashobject(ml($src,array('cache'=>$cache),true,'&'),$width,$height,
111807bf32b2SAndreas Gohr                                     array('quality' => 'high'),
111907bf32b2SAndreas Gohr                                     null,
112007bf32b2SAndreas Gohr                                     $att,
112107bf32b2SAndreas Gohr                                     $this->_xmlEntities($title));
11220f428d7dSAndreas Gohr        }elseif($title){
11233fd0b676Sandi            // well at least we have a title to display
11243fd0b676Sandi            $ret .= $this->_xmlEntities($title);
11253fd0b676Sandi        }else{
11265291ca3aSAndreas Gohr            // just show the sourcename
11273009a773SAndreas Gohr            $ret .= $this->_xmlEntities(utf8_basename(noNS($src)));
11283fd0b676Sandi        }
11293fd0b676Sandi
11303fd0b676Sandi        return $ret;
11313fd0b676Sandi    }
11323fd0b676Sandi
1133433bef32Sandi    function _xmlEntities($string) {
1134de117061Schris        return htmlspecialchars($string,ENT_QUOTES,'UTF-8');
11350cecf9d5Sandi    }
11360cecf9d5Sandi
11378a831f2bSAndreas Gohr    /**
11388a831f2bSAndreas Gohr     * Creates a linkid from a headline
1139c5a8fd96SAndreas Gohr     *
1140c5a8fd96SAndreas Gohr     * @param string  $title   The headline title
1141c5a8fd96SAndreas Gohr     * @param boolean $create  Create a new unique ID?
1142c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
11438a831f2bSAndreas Gohr     */
1144c5a8fd96SAndreas Gohr    function _headerToLink($title,$create=false) {
1145c5a8fd96SAndreas Gohr        if($create){
11464ceab83fSAndreas Gohr            return sectionID($title,$this->headers);
11474ceab83fSAndreas Gohr        }else{
1148443d207bSAndreas Gohr            $check = false;
1149443d207bSAndreas Gohr            return sectionID($title,$check);
1150c5a8fd96SAndreas Gohr        }
11510cecf9d5Sandi    }
11520cecf9d5Sandi
1153af587fa8Sandi    /**
11543fd0b676Sandi     * Construct a title and handle images in titles
11553fd0b676Sandi     *
11560b7c14c2Sandi     * @author Harry Fuecks <hfuecks@gmail.com>
11573fd0b676Sandi     */
11580ea51e63SMatt Perry    function _getLinkTitle($title, $default, & $isImage, $id=null, $linktype='content') {
1159bb0a59d4Sjan        global $conf;
1160bb0a59d4Sjan
116144881bd0Shenning.noren        $isImage = false;
116229657f9eSAndreas Gohr        if ( is_array($title) ) {
116329657f9eSAndreas Gohr            $isImage = true;
116429657f9eSAndreas Gohr            return $this->_imageTitle($title);
116529657f9eSAndreas Gohr        } elseif ( is_null($title) || trim($title)=='') {
1166fe9ec250SChris Smith            if (useHeading($linktype) && $id) {
116767c15eceSMichael Hamann                $heading = p_get_first_heading($id);
1168bb0a59d4Sjan                if ($heading) {
1169433bef32Sandi                    return $this->_xmlEntities($heading);
1170bb0a59d4Sjan                }
1171bb0a59d4Sjan            }
1172433bef32Sandi            return $this->_xmlEntities($default);
117368c26e6dSMichael Klier        } else {
117468c26e6dSMichael Klier            return $this->_xmlEntities($title);
11750cecf9d5Sandi        }
11760cecf9d5Sandi    }
11770cecf9d5Sandi
11780cecf9d5Sandi    /**
11793fd0b676Sandi     * Returns an HTML code for images used in link titles
11803fd0b676Sandi     *
11813fd0b676Sandi     * @todo Resolve namespace on internal images
11823fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
11830cecf9d5Sandi     */
1184433bef32Sandi    function _imageTitle($img) {
1185d9baf1a7SKazutaka Miyasaka        global $ID;
1186d9baf1a7SKazutaka Miyasaka
1187d9baf1a7SKazutaka Miyasaka        // some fixes on $img['src']
1188d9baf1a7SKazutaka Miyasaka        // see internalmedia() and externalmedia()
1189d9baf1a7SKazutaka Miyasaka        list($img['src'],$hash) = explode('#',$img['src'],2);
1190d9baf1a7SKazutaka Miyasaka        if ($img['type'] == 'internalmedia') {
1191d9baf1a7SKazutaka Miyasaka            resolve_mediaid(getNS($ID),$img['src'],$exists);
1192d9baf1a7SKazutaka Miyasaka        }
1193d9baf1a7SKazutaka Miyasaka
1194433bef32Sandi        return $this->_media($img['src'],
11954826ab45Sandi                              $img['title'],
11964826ab45Sandi                              $img['align'],
11974826ab45Sandi                              $img['width'],
11984826ab45Sandi                              $img['height'],
11994826ab45Sandi                              $img['cache']);
12000cecf9d5Sandi    }
1201b739ff0fSPierre Spring
1202b739ff0fSPierre Spring    /**
1203b739ff0fSPierre Spring     * _getMediaLinkConf is a helperfunction to internalmedia() and externalmedia()
1204b739ff0fSPierre Spring     * which returns a basic link to a media.
1205b739ff0fSPierre Spring     *
1206b739ff0fSPierre Spring     * @author Pierre Spring <pierre.spring@liip.ch>
1207b739ff0fSPierre Spring     * @param string $src
1208b739ff0fSPierre Spring     * @param string $title
1209b739ff0fSPierre Spring     * @param string $align
1210b739ff0fSPierre Spring     * @param string $width
1211b739ff0fSPierre Spring     * @param string $height
1212b739ff0fSPierre Spring     * @param string $cache
1213b739ff0fSPierre Spring     * @param string $render
1214b739ff0fSPierre Spring     * @access protected
1215b739ff0fSPierre Spring     * @return array
1216b739ff0fSPierre Spring     */
1217d91ab76fSMatt Perry    function _getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render) {
1218b739ff0fSPierre Spring        global $conf;
1219b739ff0fSPierre Spring
1220b739ff0fSPierre Spring        $link = array();
1221b739ff0fSPierre Spring        $link['class']  = 'media';
1222b739ff0fSPierre Spring        $link['style']  = '';
1223b739ff0fSPierre Spring        $link['pre']    = '';
1224b739ff0fSPierre Spring        $link['suf']    = '';
1225b739ff0fSPierre Spring        $link['more']   = '';
1226b739ff0fSPierre Spring        $link['target'] = $conf['target']['media'];
1227b739ff0fSPierre Spring        $link['title']  = $this->_xmlEntities($src);
1228b739ff0fSPierre Spring        $link['name']   = $this->_media($src, $title, $align, $width, $height, $cache, $render);
1229b739ff0fSPierre Spring
1230b739ff0fSPierre Spring        return $link;
1231b739ff0fSPierre Spring    }
123291459163SAnika Henke
123391459163SAnika Henke
12340cecf9d5Sandi}
12350cecf9d5Sandi
1236e3776c06SMichael Hamann//Setup VIM: ex: et ts=4 :
1237