xref: /dokuwiki/inc/parser/xhtml.php (revision f23eef27e07dfe76ab76fda68242d44de10e4022)
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     *
565*f23eef27SGerrit Uitslag     * @param string $id pageid
566*f23eef27SGerrit Uitslag     * @param string|null $name link name
567*f23eef27SGerrit Uitslag     * @param string|null $search adds search url param
568*f23eef27SGerrit Uitslag     * @param bool $returnonly whether to return html or write to doc attribute
569*f23eef27SGerrit Uitslag     * @param string $linktype type to set use of headings
570*f23eef27SGerrit Uitslag     * @return void|string writes to doc attribute or returns html depends on $returnonly
5713fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
572cffcc403Sandi     */
5730ea51e63SMatt Perry    function internallink($id, $name = null, $search=null,$returnonly=false,$linktype='content') {
574ba11bd29Sandi        global $conf;
57537e34a5eSandi        global $ID;
576c4dda6afSAnika Henke        global $INFO;
57744653a53SAdrian Lang
5783d5e07d9SAdrian Lang        $params = '';
5793d5e07d9SAdrian Lang        $parts = explode('?', $id, 2);
5803d5e07d9SAdrian Lang        if (count($parts) === 2) {
5813d5e07d9SAdrian Lang            $id = $parts[0];
5823d5e07d9SAdrian Lang            $params = $parts[1];
58344653a53SAdrian Lang        }
58444653a53SAdrian Lang
585fda14ffcSIzidor Matušov        // For empty $id we need to know the current $ID
586fda14ffcSIzidor Matušov        // We need this check because _simpleTitle needs
587fda14ffcSIzidor Matušov        // correct $id and resolve_pageid() use cleanID($id)
588fda14ffcSIzidor Matušov        // (some things could be lost)
589fda14ffcSIzidor Matušov        if ($id === '') {
590fda14ffcSIzidor Matušov            $id = $ID;
591fda14ffcSIzidor Matušov        }
592fda14ffcSIzidor Matušov
5930339c872Sjan        // default name is based on $id as given
5940339c872Sjan        $default = $this->_simpleTitle($id);
595ad32e47eSAndreas Gohr
5960339c872Sjan        // now first resolve and clean up the $id
59737e34a5eSandi        resolve_pageid(getNS($ID),$id,$exists);
598fda14ffcSIzidor Matušov
599fe9ec250SChris Smith        $name = $this->_getLinkTitle($name, $default, $isImage, $id, $linktype);
6000e1c636eSandi        if ( !$isImage ) {
6010e1c636eSandi            if ( $exists ) {
602ba11bd29Sandi                $class='wikilink1';
6030cecf9d5Sandi            } else {
604ba11bd29Sandi                $class='wikilink2';
60544a6b4c7SAndreas Gohr                $link['rel']='nofollow';
6060cecf9d5Sandi            }
6070cecf9d5Sandi        } else {
608ba11bd29Sandi            $class='media';
6090cecf9d5Sandi        }
6100cecf9d5Sandi
611a1685bedSandi        //keep hash anchor
612ce6b63d9Schris        list($id,$hash) = explode('#',$id,2);
613943dedc6SAndreas Gohr        if(!empty($hash)) $hash = $this->_headerToLink($hash);
614a1685bedSandi
615ba11bd29Sandi        //prepare for formating
616ba11bd29Sandi        $link['target'] = $conf['target']['wiki'];
617ba11bd29Sandi        $link['style']  = '';
618ba11bd29Sandi        $link['pre']    = '';
619ba11bd29Sandi        $link['suf']    = '';
62040eb54bbSjan        // highlight link to current page
621c4dda6afSAnika Henke        if ($id == $INFO['id']) {
62292795d04Sandi            $link['pre']    = '<span class="curid">';
62392795d04Sandi            $link['suf']    = '</span>';
62440eb54bbSjan        }
6255e163278SAndreas Gohr        $link['more']   = '';
626ba11bd29Sandi        $link['class']  = $class;
62744653a53SAdrian Lang        $link['url']    = wl($id, $params);
628ba11bd29Sandi        $link['name']   = $name;
629ba11bd29Sandi        $link['title']  = $id;
630723d78dbSandi        //add search string
631723d78dbSandi        if($search){
632546d3a99SAndreas Gohr            ($conf['userewrite']) ? $link['url'].='?' : $link['url'].='&amp;';
633546d3a99SAndreas Gohr            if(is_array($search)){
634546d3a99SAndreas Gohr                $search = array_map('rawurlencode',$search);
635546d3a99SAndreas Gohr                $link['url'] .= 's[]='.join('&amp;s[]=',$search);
636546d3a99SAndreas Gohr            }else{
637546d3a99SAndreas Gohr                $link['url'] .= 's='.rawurlencode($search);
638546d3a99SAndreas Gohr            }
639723d78dbSandi        }
640723d78dbSandi
641a1685bedSandi        //keep hash
642a1685bedSandi        if($hash) $link['url'].='#'.$hash;
643a1685bedSandi
644ba11bd29Sandi        //output formatted
645cffcc403Sandi        if($returnonly){
646cffcc403Sandi            return $this->_formatLink($link);
647cffcc403Sandi        }else{
648a2d649c4Sandi            $this->doc .= $this->_formatLink($link);
6490cecf9d5Sandi        }
650cffcc403Sandi    }
6510cecf9d5Sandi
6520ea51e63SMatt Perry    function externallink($url, $name = null) {
653b625487dSandi        global $conf;
6540cecf9d5Sandi
655433bef32Sandi        $name = $this->_getLinkTitle($name, $url, $isImage);
6566f0c5dbfSandi
657b52b1596SAndreas Gohr        // url might be an attack vector, only allow registered protocols
658b52b1596SAndreas Gohr        if(is_null($this->schemes)) $this->schemes = getSchemes();
659b52b1596SAndreas Gohr        list($scheme) = explode('://',$url);
660b52b1596SAndreas Gohr        $scheme = strtolower($scheme);
661b52b1596SAndreas Gohr        if(!in_array($scheme,$this->schemes)) $url = '';
662b52b1596SAndreas Gohr
663b52b1596SAndreas Gohr        // is there still an URL?
664b52b1596SAndreas Gohr        if(!$url){
665b52b1596SAndreas Gohr            $this->doc .= $name;
666b52b1596SAndreas Gohr            return;
667b52b1596SAndreas Gohr        }
668b52b1596SAndreas Gohr
669b52b1596SAndreas Gohr        // set class
6700cecf9d5Sandi        if ( !$isImage ) {
671b625487dSandi            $class='urlextern';
6720cecf9d5Sandi        } else {
673b625487dSandi            $class='media';
6740cecf9d5Sandi        }
6750cecf9d5Sandi
676b625487dSandi        //prepare for formating
677b625487dSandi        $link['target'] = $conf['target']['extern'];
678b625487dSandi        $link['style']  = '';
679b625487dSandi        $link['pre']    = '';
680b625487dSandi        $link['suf']    = '';
6815e163278SAndreas Gohr        $link['more']   = '';
682b625487dSandi        $link['class']  = $class;
683b625487dSandi        $link['url']    = $url;
684e1c10e4dSchris
685b625487dSandi        $link['name']   = $name;
686433bef32Sandi        $link['title']  = $this->_xmlEntities($url);
687b625487dSandi        if($conf['relnofollow']) $link['more'] .= ' rel="nofollow"';
6880cecf9d5Sandi
689b625487dSandi        //output formatted
690a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
6910cecf9d5Sandi    }
6920cecf9d5Sandi
6930cecf9d5Sandi    /**
6940cecf9d5Sandi    */
6950ea51e63SMatt Perry    function interwikilink($match, $name = null, $wikiName, $wikiUri) {
696b625487dSandi        global $conf;
6970cecf9d5Sandi
69897a3e4e3Sandi        $link = array();
69997a3e4e3Sandi        $link['target'] = $conf['target']['interwiki'];
70097a3e4e3Sandi        $link['pre']    = '';
70197a3e4e3Sandi        $link['suf']    = '';
7025e163278SAndreas Gohr        $link['more']   = '';
703433bef32Sandi        $link['name']   = $this->_getLinkTitle($name, $wikiUri, $isImage);
7040cecf9d5Sandi
70597a3e4e3Sandi        //get interwiki URL
7061f82fabeSAndreas Gohr        $url = $this->_resolveInterWiki($wikiName,$wikiUri);
7070cecf9d5Sandi
70897a3e4e3Sandi        if ( !$isImage ) {
7099d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$wikiName);
7109d2ddea4SAndreas Gohr            $link['class'] = "interwiki iw_$class";
7111c2d1019SAndreas Gohr        } else {
7121c2d1019SAndreas Gohr            $link['class'] = 'media';
71397a3e4e3Sandi        }
7140cecf9d5Sandi
71597a3e4e3Sandi        //do we stay at the same server? Use local target
71697a3e4e3Sandi        if( strpos($url,DOKU_URL) === 0 ){
71797a3e4e3Sandi            $link['target'] = $conf['target']['wiki'];
71897a3e4e3Sandi        }
7190cecf9d5Sandi
72097a3e4e3Sandi        $link['url'] = $url;
72197a3e4e3Sandi        $link['title'] = htmlspecialchars($link['url']);
72297a3e4e3Sandi
72397a3e4e3Sandi        //output formatted
724a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7250cecf9d5Sandi    }
7260cecf9d5Sandi
7270cecf9d5Sandi    /**
7280cecf9d5Sandi     */
7290ea51e63SMatt Perry    function windowssharelink($url, $name = null) {
7301d47afe1Sandi        global $conf;
7311d47afe1Sandi        global $lang;
7321d47afe1Sandi        //simple setup
7331d47afe1Sandi        $link['target'] = $conf['target']['windows'];
7341d47afe1Sandi        $link['pre']    = '';
7351d47afe1Sandi        $link['suf']   = '';
7361d47afe1Sandi        $link['style']  = '';
7370cecf9d5Sandi
738433bef32Sandi        $link['name'] = $this->_getLinkTitle($name, $url, $isImage);
7390cecf9d5Sandi        if ( !$isImage ) {
7401d47afe1Sandi            $link['class'] = 'windows';
7410cecf9d5Sandi        } else {
7421d47afe1Sandi            $link['class'] = 'media';
7430cecf9d5Sandi        }
7440cecf9d5Sandi
745433bef32Sandi        $link['title'] = $this->_xmlEntities($url);
7461d47afe1Sandi        $url = str_replace('\\','/',$url);
7471d47afe1Sandi        $url = 'file:///'.$url;
7481d47afe1Sandi        $link['url'] = $url;
7490cecf9d5Sandi
7501d47afe1Sandi        //output formatted
751a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7520cecf9d5Sandi    }
7530cecf9d5Sandi
7540ea51e63SMatt Perry    function emaillink($address, $name = null) {
75571352defSandi        global $conf;
75671352defSandi        //simple setup
75771352defSandi        $link = array();
75871352defSandi        $link['target'] = '';
75971352defSandi        $link['pre']    = '';
76071352defSandi        $link['suf']   = '';
76171352defSandi        $link['style']  = '';
76271352defSandi        $link['more']   = '';
7630cecf9d5Sandi
764c078fc55SAndreas Gohr        $name = $this->_getLinkTitle($name, '', $isImage);
7650cecf9d5Sandi        if ( !$isImage ) {
766be96545cSAnika Henke            $link['class']='mail';
7670cecf9d5Sandi        } else {
768be96545cSAnika Henke            $link['class']='media';
7690cecf9d5Sandi        }
7700cecf9d5Sandi
77107738714SAndreas Gohr        $address = $this->_xmlEntities($address);
77200a7b5adSEsther Brunner        $address = obfuscate($address);
77300a7b5adSEsther Brunner        $title   = $address;
7748c128049SAndreas Gohr
77571352defSandi        if(empty($name)){
77600a7b5adSEsther Brunner            $name = $address;
77771352defSandi        }
7780cecf9d5Sandi
779776b36ecSAndreas Gohr        if($conf['mailguard'] == 'visible') $address = rawurlencode($address);
780776b36ecSAndreas Gohr
781776b36ecSAndreas Gohr        $link['url']   = 'mailto:'.$address;
78271352defSandi        $link['name']  = $name;
78371352defSandi        $link['title'] = $title;
7840cecf9d5Sandi
78571352defSandi        //output formatted
786a2d649c4Sandi        $this->doc .= $this->_formatLink($link);
7870cecf9d5Sandi    }
7880cecf9d5Sandi
7890ea51e63SMatt Perry    function internalmedia ($src, $title=null, $align=null, $width=null,
790c5393ecbSAnika Henke                            $height=null, $cache=null, $linking=null, $return=NULL) {
79137e34a5eSandi        global $ID;
79291df343aSAndreas Gohr        list($src,$hash) = explode('#',$src,2);
79337e34a5eSandi        resolve_mediaid(getNS($ID),$src, $exists);
7940cecf9d5Sandi
795d98d4540SBen Coburn        $noLink = false;
7968acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
797b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
7983685f775Sandi
79927bf7924STom N Harris        list($ext,$mime,$dl) = mimetype($src,false);
800b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
801dc673a5bSjoe.lapp            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),($linking=='direct'));
802f50634f0SAnika Henke        }elseif(($mime == 'application/x-shockwave-flash' || media_supportedav($mime)) && $render){
8032a2a2ba2SAnika Henke            // don't link movies
80444881bd0Shenning.noren            $noLink = true;
80555efc227SAndreas Gohr        }else{
8062ca14335SEsther Brunner            // add file icons
8079d2ddea4SAndreas Gohr            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
8089d2ddea4SAndreas Gohr            $link['class'] .= ' mediafile mf_'.$class;
8096de3759aSAndreas Gohr            $link['url'] = ml($src,array('id'=>$ID,'cache'=>$cache),true);
81091328684SMichael Hamann            if ($exists) $link['title'] .= ' (' . filesize_h(filesize(mediaFN($src))).')';
81155efc227SAndreas Gohr        }
8123685f775Sandi
81391df343aSAndreas Gohr        if($hash) $link['url'] .= '#'.$hash;
81491df343aSAndreas Gohr
8156fe20453SGina Haeussge        //markup non existing files
8164a24b459SKate Arzamastseva        if (!$exists) {
8176fe20453SGina Haeussge            $link['class'] .= ' wikilink2';
8184a24b459SKate Arzamastseva        }
8196fe20453SGina Haeussge
8203685f775Sandi        //output formatted
821f50634f0SAnika Henke        if ($return) {
822f50634f0SAnika Henke            if ($linking == 'nolink' || $noLink) return $link['name'];
823f50634f0SAnika Henke            else return $this->_formatLink($link);
824f50634f0SAnika Henke        } else {
825dc673a5bSjoe.lapp            if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
8262ca14335SEsther Brunner            else $this->doc .= $this->_formatLink($link);
8270cecf9d5Sandi        }
828f50634f0SAnika Henke    }
8290cecf9d5Sandi
8300ea51e63SMatt Perry    function externalmedia ($src, $title=null, $align=null, $width=null,
8310ea51e63SMatt Perry                            $height=null, $cache=null, $linking=null) {
83291df343aSAndreas Gohr        list($src,$hash) = explode('#',$src,2);
833d98d4540SBen Coburn        $noLink = false;
8348acb3108SAndreas Gohr        $render = ($linking == 'linkonly') ? false : true;
835b739ff0fSPierre Spring        $link = $this->_getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render);
836b739ff0fSPierre Spring
837b739ff0fSPierre Spring        $link['url']    = ml($src,array('cache'=>$cache));
8383685f775Sandi
83927bf7924STom N Harris        list($ext,$mime,$dl) = mimetype($src,false);
840b739ff0fSPierre Spring        if(substr($mime,0,5) == 'image' && $render){
8412ca14335SEsther Brunner            // link only jpeg images
84244881bd0Shenning.noren            // if ($ext != 'jpg' && $ext != 'jpeg') $noLink = true;
843f50634f0SAnika Henke        }elseif(($mime == 'application/x-shockwave-flash' || media_supportedav($mime)) && $render){
8442a2a2ba2SAnika Henke            // don't link movies
84544881bd0Shenning.noren            $noLink = true;
8462ca14335SEsther Brunner        }else{
8472ca14335SEsther Brunner            // add file icons
84827bf7924STom N Harris            $class = preg_replace('/[^_\-a-z0-9]+/i','_',$ext);
84927bf7924STom N Harris            $link['class'] .= ' mediafile mf_'.$class;
8502ca14335SEsther Brunner        }
8512ca14335SEsther Brunner
85291df343aSAndreas Gohr        if($hash) $link['url'] .= '#'.$hash;
85391df343aSAndreas Gohr
8543685f775Sandi        //output formatted
855dc673a5bSjoe.lapp        if ($linking == 'nolink' || $noLink) $this->doc .= $link['name'];
8562ca14335SEsther Brunner        else $this->doc .= $this->_formatLink($link);
8570cecf9d5Sandi    }
8580cecf9d5Sandi
8594826ab45Sandi    /**
8603db95becSAndreas Gohr     * Renders an RSS feed
861b625487dSandi     *
862b625487dSandi     * @author Andreas Gohr <andi@splitbrain.org>
863b625487dSandi     */
8643db95becSAndreas Gohr    function rss ($url,$params){
865b625487dSandi        global $lang;
8663db95becSAndreas Gohr        global $conf;
8673db95becSAndreas Gohr
8683db95becSAndreas Gohr        require_once(DOKU_INC.'inc/FeedParser.php');
8693db95becSAndreas Gohr        $feed = new FeedParser();
87000077af8SAndreas Gohr        $feed->set_feed_url($url);
871b625487dSandi
872b625487dSandi        //disable warning while fetching
873bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { $elvl = error_reporting(E_ERROR); }
8743db95becSAndreas Gohr        $rc = $feed->init();
875bad905f1SBen Coburn        if (!defined('DOKU_E_LEVEL')) { error_reporting($elvl); }
876b625487dSandi
8773db95becSAndreas Gohr        //decide on start and end
8783db95becSAndreas Gohr        if($params['reverse']){
8793db95becSAndreas Gohr            $mod = -1;
8803db95becSAndreas Gohr            $start = $feed->get_item_quantity()-1;
8813db95becSAndreas Gohr            $end   = $start - ($params['max']);
882b2a412b0SAndreas Gohr            $end   = ($end < -1) ? -1 : $end;
8833db95becSAndreas Gohr        }else{
8843db95becSAndreas Gohr            $mod   = 1;
8853db95becSAndreas Gohr            $start = 0;
8863db95becSAndreas Gohr            $end   = $feed->get_item_quantity();
887d91ab76fSMatt Perry            $end   = ($end > $params['max']) ? $params['max'] : $end;
8883db95becSAndreas Gohr        }
8893db95becSAndreas Gohr
890a2d649c4Sandi        $this->doc .= '<ul class="rss">';
8913db95becSAndreas Gohr        if($rc){
8923db95becSAndreas Gohr            for ($x = $start; $x != $end; $x += $mod) {
8931bde1582SAndreas Gohr                $item = $feed->get_item($x);
8943db95becSAndreas Gohr                $this->doc .= '<li><div class="li">';
895d2ea3363SAndreas Gohr                // support feeds without links
896d2ea3363SAndreas Gohr                $lnkurl = $item->get_permalink();
897d2ea3363SAndreas Gohr                if($lnkurl){
898793361f8SAndreas Gohr                    // title is escaped by SimplePie, we unescape here because it
899793361f8SAndreas Gohr                    // is escaped again in externallink() FS#1705
9001bde1582SAndreas Gohr                    $this->externallink($item->get_permalink(),
901b9b9b28bSMichael Hamann                                        html_entity_decode($item->get_title(), ENT_QUOTES, 'UTF-8'));
902d2ea3363SAndreas Gohr                }else{
903d2ea3363SAndreas Gohr                    $this->doc .= ' '.$item->get_title();
904d2ea3363SAndreas Gohr                }
9053db95becSAndreas Gohr                if($params['author']){
9061bde1582SAndreas Gohr                    $author = $item->get_author(0);
9071bde1582SAndreas Gohr                    if($author){
9081bde1582SAndreas Gohr                        $name = $author->get_name();
9091bde1582SAndreas Gohr                        if(!$name) $name = $author->get_email();
9101bde1582SAndreas Gohr                        if($name) $this->doc .= ' '.$lang['by'].' '.$name;
9111bde1582SAndreas Gohr                    }
9123db95becSAndreas Gohr                }
9133db95becSAndreas Gohr                if($params['date']){
9142e7e0c29SAndreas Gohr                    $this->doc .= ' ('.$item->get_local_date($conf['dformat']).')';
9153db95becSAndreas Gohr                }
9161bde1582SAndreas Gohr                if($params['details']){
9173db95becSAndreas Gohr                    $this->doc .= '<div class="detail">';
918173dccb7STom N Harris                    if($conf['htmlok']){
9191bde1582SAndreas Gohr                        $this->doc .= $item->get_description();
9203db95becSAndreas Gohr                    }else{
9211bde1582SAndreas Gohr                        $this->doc .= strip_tags($item->get_description());
9223db95becSAndreas Gohr                    }
9233db95becSAndreas Gohr                    $this->doc .= '</div>';
9243db95becSAndreas Gohr                }
9253db95becSAndreas Gohr
9263db95becSAndreas Gohr                $this->doc .= '</div></li>';
927b625487dSandi            }
928b625487dSandi        }else{
9293db95becSAndreas Gohr            $this->doc .= '<li><div class="li">';
930a2d649c4Sandi            $this->doc .= '<em>'.$lang['rssfailed'].'</em>';
931b625487dSandi            $this->externallink($url);
93245e147ccSAndreas Gohr            if($conf['allowdebug']){
93345e147ccSAndreas Gohr                $this->doc .= '<!--'.hsc($feed->error).'-->';
93445e147ccSAndreas Gohr            }
9353db95becSAndreas Gohr            $this->doc .= '</div></li>';
936b625487dSandi        }
937a2d649c4Sandi        $this->doc .= '</ul>';
938b625487dSandi    }
939b625487dSandi
9400cecf9d5Sandi    // $numrows not yet implemented
941619736fdSAdrian Lang    function table_open($maxcols = null, $numrows = null, $pos = null){
94290df9a4dSAdrian Lang        global $lang;
943b5742cedSPierre Spring        // initialize the row counter used for classes
944b5742cedSPierre Spring        $this->_counter['row_counter'] = 0;
945619736fdSAdrian Lang        $class = 'table';
946619736fdSAdrian Lang        if ($pos !== null) {
947619736fdSAdrian Lang            $class .= ' ' . $this->startSectionEdit($pos, 'table');
948619736fdSAdrian Lang        }
949619736fdSAdrian Lang        $this->doc .= '<div class="' . $class . '"><table class="inline">' .
950619736fdSAdrian Lang                      DOKU_LF;
9510cecf9d5Sandi    }
9520cecf9d5Sandi
953619736fdSAdrian Lang    function table_close($pos = null){
954a8574918SAnika Henke        $this->doc .= '</table></div>'.DOKU_LF;
955619736fdSAdrian Lang        if ($pos !== null) {
95690df9a4dSAdrian Lang            $this->finishSectionEdit($pos);
9570cecf9d5Sandi        }
958619736fdSAdrian Lang    }
9590cecf9d5Sandi
9600cecf9d5Sandi    function tablerow_open(){
961b5742cedSPierre Spring        // initialize the cell counter used for classes
962b5742cedSPierre Spring        $this->_counter['cell_counter'] = 0;
963b5742cedSPierre Spring        $class = 'row' . $this->_counter['row_counter']++;
964b5742cedSPierre Spring        $this->doc .= DOKU_TAB . '<tr class="'.$class.'">' . DOKU_LF . DOKU_TAB . DOKU_TAB;
9650cecf9d5Sandi    }
9660cecf9d5Sandi
9670cecf9d5Sandi    function tablerow_close(){
968a2d649c4Sandi        $this->doc .= DOKU_LF . DOKU_TAB . '</tr>' . DOKU_LF;
9690cecf9d5Sandi    }
9700cecf9d5Sandi
9710ea51e63SMatt Perry    function tableheader_open($colspan = 1, $align = null, $rowspan = 1){
972b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
9730cecf9d5Sandi        if ( !is_null($align) ) {
974b5742cedSPierre Spring            $class .= ' '.$align.'align';
9750cecf9d5Sandi        }
976b5742cedSPierre Spring        $class .= '"';
977b5742cedSPierre Spring        $this->doc .= '<th ' . $class;
9780cecf9d5Sandi        if ( $colspan > 1 ) {
979a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
980a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
9810cecf9d5Sandi        }
98225b97867Shakan.sandell        if ( $rowspan > 1 ) {
98325b97867Shakan.sandell            $this->doc .= ' rowspan="'.$rowspan.'"';
98425b97867Shakan.sandell        }
985a2d649c4Sandi        $this->doc .= '>';
9860cecf9d5Sandi    }
9870cecf9d5Sandi
9880cecf9d5Sandi    function tableheader_close(){
989a2d649c4Sandi        $this->doc .= '</th>';
9900cecf9d5Sandi    }
9910cecf9d5Sandi
9920ea51e63SMatt Perry    function tablecell_open($colspan = 1, $align = null, $rowspan = 1){
993b5742cedSPierre Spring        $class = 'class="col' . $this->_counter['cell_counter']++;
9940cecf9d5Sandi        if ( !is_null($align) ) {
995b5742cedSPierre Spring            $class .= ' '.$align.'align';
9960cecf9d5Sandi        }
997b5742cedSPierre Spring        $class .= '"';
998b5742cedSPierre Spring        $this->doc .= '<td '.$class;
9990cecf9d5Sandi        if ( $colspan > 1 ) {
1000a28fd914SAndreas Gohr            $this->_counter['cell_counter'] += $colspan-1;
1001a2d649c4Sandi            $this->doc .= ' colspan="'.$colspan.'"';
10020cecf9d5Sandi        }
100325b97867Shakan.sandell        if ( $rowspan > 1 ) {
100425b97867Shakan.sandell            $this->doc .= ' rowspan="'.$rowspan.'"';
100525b97867Shakan.sandell        }
1006a2d649c4Sandi        $this->doc .= '>';
10070cecf9d5Sandi    }
10080cecf9d5Sandi
10090cecf9d5Sandi    function tablecell_close(){
1010a2d649c4Sandi        $this->doc .= '</td>';
10110cecf9d5Sandi    }
10120cecf9d5Sandi
10130cecf9d5Sandi    //----------------------------------------------------------
10140cecf9d5Sandi    // Utils
10150cecf9d5Sandi
1016ba11bd29Sandi    /**
10173fd0b676Sandi     * Build a link
10183fd0b676Sandi     *
10193fd0b676Sandi     * Assembles all parts defined in $link returns HTML for the link
1020ba11bd29Sandi     *
1021ba11bd29Sandi     * @author Andreas Gohr <andi@splitbrain.org>
1022ba11bd29Sandi     */
1023433bef32Sandi    function _formatLink($link){
1024ba11bd29Sandi        //make sure the url is XHTML compliant (skip mailto)
1025ba11bd29Sandi        if(substr($link['url'],0,7) != 'mailto:'){
1026ba11bd29Sandi            $link['url'] = str_replace('&','&amp;',$link['url']);
1027ba11bd29Sandi            $link['url'] = str_replace('&amp;amp;','&amp;',$link['url']);
1028ba11bd29Sandi        }
1029ba11bd29Sandi        //remove double encodings in titles
1030ba11bd29Sandi        $link['title'] = str_replace('&amp;amp;','&amp;',$link['title']);
1031ba11bd29Sandi
1032453493f2SAndreas Gohr        // be sure there are no bad chars in url or title
1033453493f2SAndreas Gohr        // (we can't do this for name because it can contain an img tag)
1034453493f2SAndreas Gohr        $link['url']   = strtr($link['url'],array('>'=>'%3E','<'=>'%3C','"'=>'%22'));
1035453493f2SAndreas Gohr        $link['title'] = strtr($link['title'],array('>'=>'&gt;','<'=>'&lt;','"'=>'&quot;'));
1036453493f2SAndreas Gohr
1037ba11bd29Sandi        $ret  = '';
1038ba11bd29Sandi        $ret .= $link['pre'];
1039ba11bd29Sandi        $ret .= '<a href="'.$link['url'].'"';
1040bb4866bdSchris        if(!empty($link['class']))  $ret .= ' class="'.$link['class'].'"';
1041bb4866bdSchris        if(!empty($link['target'])) $ret .= ' target="'.$link['target'].'"';
1042bb4866bdSchris        if(!empty($link['title']))  $ret .= ' title="'.$link['title'].'"';
1043bb4866bdSchris        if(!empty($link['style']))  $ret .= ' style="'.$link['style'].'"';
104444a6b4c7SAndreas Gohr        if(!empty($link['rel']))    $ret .= ' rel="'.$link['rel'].'"';
1045bb4866bdSchris        if(!empty($link['more']))   $ret .= ' '.$link['more'];
1046ba11bd29Sandi        $ret .= '>';
1047ba11bd29Sandi        $ret .= $link['name'];
1048ba11bd29Sandi        $ret .= '</a>';
1049ba11bd29Sandi        $ret .= $link['suf'];
1050ba11bd29Sandi        return $ret;
1051ba11bd29Sandi    }
1052ba11bd29Sandi
1053ba11bd29Sandi    /**
10543fd0b676Sandi     * Renders internal and external media
10553fd0b676Sandi     *
10563fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
10573fd0b676Sandi     */
10580ea51e63SMatt Perry    function _media ($src, $title=null, $align=null, $width=null,
10590ea51e63SMatt Perry                      $height=null, $cache=null, $render = true) {
10603fd0b676Sandi
10613fd0b676Sandi        $ret = '';
10623fd0b676Sandi
1063ecebf3a8SAndreas Gohr        list($ext,$mime,$dl) = mimetype($src);
10643fd0b676Sandi        if(substr($mime,0,5) == 'image'){
1065b739ff0fSPierre Spring            // first get the $title
1066b739ff0fSPierre Spring            if (!is_null($title)) {
1067b739ff0fSPierre Spring                $title  = $this->_xmlEntities($title);
1068b739ff0fSPierre Spring            }elseif($ext == 'jpg' || $ext == 'jpeg'){
1069b739ff0fSPierre Spring                //try to use the caption from IPTC/EXIF
1070b739ff0fSPierre Spring                require_once(DOKU_INC.'inc/JpegMeta.php');
107167f9913dSAndreas Gohr                $jpeg =new JpegMeta(mediaFN($src));
1072b739ff0fSPierre Spring                if($jpeg !== false) $cap = $jpeg->getTitle();
1073b739ff0fSPierre Spring                if($cap){
1074b739ff0fSPierre Spring                    $title = $this->_xmlEntities($cap);
1075b739ff0fSPierre Spring                }
1076b739ff0fSPierre Spring            }
1077b739ff0fSPierre Spring            if (!$render) {
1078b739ff0fSPierre Spring                // if the picture is not supposed to be rendered
1079b739ff0fSPierre Spring                // return the title of the picture
1080b739ff0fSPierre Spring                if (!$title) {
1081b739ff0fSPierre Spring                    // just show the sourcename
10823009a773SAndreas Gohr                    $title = $this->_xmlEntities(utf8_basename(noNS($src)));
1083b739ff0fSPierre Spring                }
1084b739ff0fSPierre Spring                return $title;
1085b739ff0fSPierre Spring            }
10863fd0b676Sandi            //add image tag
10876de3759aSAndreas Gohr            $ret .= '<img src="'.ml($src,array('w'=>$width,'h'=>$height,'cache'=>$cache)).'"';
10883fd0b676Sandi            $ret .= ' class="media'.$align.'"';
10893fd0b676Sandi
1090b739ff0fSPierre Spring            if ($title) {
1091b739ff0fSPierre Spring                $ret .= ' title="' . $title . '"';
1092b739ff0fSPierre Spring                $ret .= ' alt="'   . $title .'"';
10933fd0b676Sandi            }else{
10943fd0b676Sandi                $ret .= ' alt=""';
10953fd0b676Sandi            }
10963fd0b676Sandi
10973fd0b676Sandi            if ( !is_null($width) )
10983fd0b676Sandi                $ret .= ' width="'.$this->_xmlEntities($width).'"';
10993fd0b676Sandi
11003fd0b676Sandi            if ( !is_null($height) )
11013fd0b676Sandi                $ret .= ' height="'.$this->_xmlEntities($height).'"';
11023fd0b676Sandi
11033fd0b676Sandi            $ret .= ' />';
11043fd0b676Sandi
110517954bb5SAnika Henke        }elseif(media_supportedav($mime, 'video') || media_supportedav($mime, 'audio')){
11062a2a2ba2SAnika Henke            // first get the $title
110717954bb5SAnika Henke            $title = !is_null($title) ? $this->_xmlEntities($title) : false;
11082a2a2ba2SAnika Henke            if (!$render) {
110917954bb5SAnika Henke                // if the file is not supposed to be rendered
111017954bb5SAnika Henke                // return the title of the file (just the sourcename if there is no title)
111117954bb5SAnika Henke                return $title ? $title : $this->_xmlEntities(utf8_basename(noNS($src)));
11122a2a2ba2SAnika Henke            }
11132a2a2ba2SAnika Henke
11142a2a2ba2SAnika Henke            $att = array();
11152a2a2ba2SAnika Henke            $att['class'] = "media$align";
111617954bb5SAnika Henke            if ($title) {
111717954bb5SAnika Henke                $att['title'] = $title;
111817954bb5SAnika Henke            }
11192a2a2ba2SAnika Henke
112017954bb5SAnika Henke            if (media_supportedav($mime, 'video')) {
112117954bb5SAnika Henke                //add video
112279e53fe5SAnika Henke                $ret .= $this->_video($src, $width, $height, $att);
1123b44a5dceSAnika Henke            }
112417954bb5SAnika Henke            if (media_supportedav($mime, 'audio')) {
1125b44a5dceSAnika Henke                //add audio
1126b44a5dceSAnika Henke                $ret .= $this->_audio($src, $att);
112717954bb5SAnika Henke            }
1128b44a5dceSAnika Henke
11293fd0b676Sandi        }elseif($mime == 'application/x-shockwave-flash'){
11301c882ba8SAndreas Gohr            if (!$render) {
11311c882ba8SAndreas Gohr                // if the flash is not supposed to be rendered
11321c882ba8SAndreas Gohr                // return the title of the flash
11331c882ba8SAndreas Gohr                if (!$title) {
11341c882ba8SAndreas Gohr                    // just show the sourcename
11353009a773SAndreas Gohr                    $title = utf8_basename(noNS($src));
11361c882ba8SAndreas Gohr                }
113707bf32b2SAndreas Gohr                return $this->_xmlEntities($title);
11381c882ba8SAndreas Gohr            }
11391c882ba8SAndreas Gohr
114007bf32b2SAndreas Gohr            $att = array();
114107bf32b2SAndreas Gohr            $att['class'] = "media$align";
114207bf32b2SAndreas Gohr            if($align == 'right') $att['align'] = 'right';
114307bf32b2SAndreas Gohr            if($align == 'left')  $att['align'] = 'left';
1144c471e6a6SAndreas Gohr            $ret .= html_flashobject(ml($src,array('cache'=>$cache),true,'&'),$width,$height,
114507bf32b2SAndreas Gohr                                     array('quality' => 'high'),
114607bf32b2SAndreas Gohr                                     null,
114707bf32b2SAndreas Gohr                                     $att,
114807bf32b2SAndreas Gohr                                     $this->_xmlEntities($title));
11490f428d7dSAndreas Gohr        }elseif($title){
11503fd0b676Sandi            // well at least we have a title to display
11513fd0b676Sandi            $ret .= $this->_xmlEntities($title);
11523fd0b676Sandi        }else{
11535291ca3aSAndreas Gohr            // just show the sourcename
11543009a773SAndreas Gohr            $ret .= $this->_xmlEntities(utf8_basename(noNS($src)));
11553fd0b676Sandi        }
11563fd0b676Sandi
11573fd0b676Sandi        return $ret;
11583fd0b676Sandi    }
11593fd0b676Sandi
1160433bef32Sandi    function _xmlEntities($string) {
1161de117061Schris        return htmlspecialchars($string,ENT_QUOTES,'UTF-8');
11620cecf9d5Sandi    }
11630cecf9d5Sandi
11648a831f2bSAndreas Gohr    /**
11658a831f2bSAndreas Gohr     * Creates a linkid from a headline
1166c5a8fd96SAndreas Gohr     *
1167c5a8fd96SAndreas Gohr     * @param string  $title   The headline title
1168c5a8fd96SAndreas Gohr     * @param boolean $create  Create a new unique ID?
1169c5a8fd96SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
11708a831f2bSAndreas Gohr     */
1171c5a8fd96SAndreas Gohr    function _headerToLink($title,$create=false) {
1172c5a8fd96SAndreas Gohr        if($create){
11734ceab83fSAndreas Gohr            return sectionID($title,$this->headers);
11744ceab83fSAndreas Gohr        }else{
1175443d207bSAndreas Gohr            $check = false;
1176443d207bSAndreas Gohr            return sectionID($title,$check);
1177c5a8fd96SAndreas Gohr        }
11780cecf9d5Sandi    }
11790cecf9d5Sandi
1180af587fa8Sandi    /**
11813fd0b676Sandi     * Construct a title and handle images in titles
11823fd0b676Sandi     *
11830b7c14c2Sandi     * @author Harry Fuecks <hfuecks@gmail.com>
11843fd0b676Sandi     */
11850ea51e63SMatt Perry    function _getLinkTitle($title, $default, & $isImage, $id=null, $linktype='content') {
1186bb0a59d4Sjan        global $conf;
1187bb0a59d4Sjan
118844881bd0Shenning.noren        $isImage = false;
118929657f9eSAndreas Gohr        if ( is_array($title) ) {
119029657f9eSAndreas Gohr            $isImage = true;
119129657f9eSAndreas Gohr            return $this->_imageTitle($title);
119229657f9eSAndreas Gohr        } elseif ( is_null($title) || trim($title)=='') {
1193fe9ec250SChris Smith            if (useHeading($linktype) && $id) {
119467c15eceSMichael Hamann                $heading = p_get_first_heading($id);
1195bb0a59d4Sjan                if ($heading) {
1196433bef32Sandi                    return $this->_xmlEntities($heading);
1197bb0a59d4Sjan                }
1198bb0a59d4Sjan            }
1199433bef32Sandi            return $this->_xmlEntities($default);
120068c26e6dSMichael Klier        } else {
120168c26e6dSMichael Klier            return $this->_xmlEntities($title);
12020cecf9d5Sandi        }
12030cecf9d5Sandi    }
12040cecf9d5Sandi
12050cecf9d5Sandi    /**
12063fd0b676Sandi     * Returns an HTML code for images used in link titles
12073fd0b676Sandi     *
12083fd0b676Sandi     * @todo Resolve namespace on internal images
12093fd0b676Sandi     * @author Andreas Gohr <andi@splitbrain.org>
12100cecf9d5Sandi     */
1211433bef32Sandi    function _imageTitle($img) {
1212d9baf1a7SKazutaka Miyasaka        global $ID;
1213d9baf1a7SKazutaka Miyasaka
1214d9baf1a7SKazutaka Miyasaka        // some fixes on $img['src']
1215d9baf1a7SKazutaka Miyasaka        // see internalmedia() and externalmedia()
1216d9baf1a7SKazutaka Miyasaka        list($img['src'],$hash) = explode('#',$img['src'],2);
1217d9baf1a7SKazutaka Miyasaka        if ($img['type'] == 'internalmedia') {
1218d9baf1a7SKazutaka Miyasaka            resolve_mediaid(getNS($ID),$img['src'],$exists);
1219d9baf1a7SKazutaka Miyasaka        }
1220d9baf1a7SKazutaka Miyasaka
1221433bef32Sandi        return $this->_media($img['src'],
12224826ab45Sandi                              $img['title'],
12234826ab45Sandi                              $img['align'],
12244826ab45Sandi                              $img['width'],
12254826ab45Sandi                              $img['height'],
12264826ab45Sandi                              $img['cache']);
12270cecf9d5Sandi    }
1228b739ff0fSPierre Spring
1229b739ff0fSPierre Spring    /**
1230b739ff0fSPierre Spring     * _getMediaLinkConf is a helperfunction to internalmedia() and externalmedia()
1231b739ff0fSPierre Spring     * which returns a basic link to a media.
1232b739ff0fSPierre Spring     *
1233b739ff0fSPierre Spring     * @author Pierre Spring <pierre.spring@liip.ch>
1234b739ff0fSPierre Spring     * @param string $src
1235b739ff0fSPierre Spring     * @param string $title
1236b739ff0fSPierre Spring     * @param string $align
1237b739ff0fSPierre Spring     * @param string $width
1238b739ff0fSPierre Spring     * @param string $height
1239b739ff0fSPierre Spring     * @param string $cache
1240b739ff0fSPierre Spring     * @param string $render
1241b739ff0fSPierre Spring     * @access protected
1242b739ff0fSPierre Spring     * @return array
1243b739ff0fSPierre Spring     */
1244d91ab76fSMatt Perry    function _getMediaLinkConf($src, $title, $align, $width, $height, $cache, $render) {
1245b739ff0fSPierre Spring        global $conf;
1246b739ff0fSPierre Spring
1247b739ff0fSPierre Spring        $link = array();
1248b739ff0fSPierre Spring        $link['class']  = 'media';
1249b739ff0fSPierre Spring        $link['style']  = '';
1250b739ff0fSPierre Spring        $link['pre']    = '';
1251b739ff0fSPierre Spring        $link['suf']    = '';
1252b739ff0fSPierre Spring        $link['more']   = '';
1253b739ff0fSPierre Spring        $link['target'] = $conf['target']['media'];
1254b739ff0fSPierre Spring        $link['title']  = $this->_xmlEntities($src);
1255b739ff0fSPierre Spring        $link['name']   = $this->_media($src, $title, $align, $width, $height, $cache, $render);
1256b739ff0fSPierre Spring
1257b739ff0fSPierre Spring        return $link;
1258b739ff0fSPierre Spring    }
125991459163SAnika Henke
126091459163SAnika Henke
12612a2a2ba2SAnika Henke    /**
12622a2a2ba2SAnika Henke     * Embed video(s) in HTML
12632a2a2ba2SAnika Henke     *
12642a2a2ba2SAnika Henke     * @author Anika Henke <anika@selfthinker.org>
12652a2a2ba2SAnika Henke     *
12662a2a2ba2SAnika Henke     * @param string $src      - ID of video to embed
12672a2a2ba2SAnika Henke     * @param int $width       - width of the video in pixels
12682a2a2ba2SAnika Henke     * @param int $height      - height of the video in pixels
12692a2a2ba2SAnika Henke     * @param array $atts      - additional attributes for the <video> tag
1270f50634f0SAnika Henke     * @return string
12712a2a2ba2SAnika Henke     */
127279e53fe5SAnika Henke    function _video($src,$width,$height,$atts=null){
12732a2a2ba2SAnika Henke        // prepare width and height
12742a2a2ba2SAnika Henke        if(is_null($atts)) $atts = array();
12752a2a2ba2SAnika Henke        $atts['width']  = (int) $width;
12762a2a2ba2SAnika Henke        $atts['height'] = (int) $height;
12772a2a2ba2SAnika Henke        if(!$atts['width'])  $atts['width']  = 320;
12782a2a2ba2SAnika Henke        if(!$atts['height']) $atts['height'] = 240;
12792a2a2ba2SAnika Henke
12803d7a9e0aSAnika Henke        // prepare alternative formats
12813d7a9e0aSAnika Henke        $extensions = array('webm', 'ogv', 'mp4');
12823d7a9e0aSAnika Henke        $alternatives = media_alternativefiles($src, $extensions);
128399f943f6SAnika Henke        $poster = media_alternativefiles($src, array('jpg', 'png'), true);
128499f943f6SAnika Henke        $posterUrl = '';
128599f943f6SAnika Henke        if (!empty($poster)) {
128699f943f6SAnika Henke            $posterUrl = ml(reset($poster),array('cache'=>$cache),true,'&');
128799f943f6SAnika Henke        }
12882a2a2ba2SAnika Henke
1289f50634f0SAnika Henke        $out = '';
129079e53fe5SAnika Henke        // open video tag
1291f50634f0SAnika Henke        $out .= '<video '.buildAttributes($atts).' controls="controls"';
12923641199aSAnika Henke        if ($posterUrl) $out .= ' poster="'.hsc($posterUrl).'"';
1293f50634f0SAnika Henke        $out .= '>'.NL;
12943641199aSAnika Henke        $fallback = '';
129579e53fe5SAnika Henke
129679e53fe5SAnika Henke        // output source for each alternative video format
129779e53fe5SAnika Henke        foreach($alternatives as $mime => $file) {
12983d7a9e0aSAnika Henke            $url = ml($file,array('cache'=>$cache),true,'&');
129917954bb5SAnika Henke            $title = $atts['title'] ? $atts['title'] : $this->_xmlEntities(utf8_basename(noNS($file)));
13003d7a9e0aSAnika Henke
1301f50634f0SAnika Henke            $out .= '<source src="'.hsc($url).'" type="'.$mime.'" />'.NL;
130279e53fe5SAnika Henke            // alternative content (just a link to the file)
13033641199aSAnika Henke            $fallback .= $this->internalmedia($file, $title, NULL, NULL, NULL, $cache=NULL, $linking='linkonly', $return=true);
13043d7a9e0aSAnika Henke        }
13052a2a2ba2SAnika Henke
13062a2a2ba2SAnika Henke        // finish
13073641199aSAnika Henke        $out .= $fallback;
1308f50634f0SAnika Henke        $out .= '</video>'.NL;
1309f50634f0SAnika Henke        return $out;
13102a2a2ba2SAnika Henke    }
13112a2a2ba2SAnika Henke
1312b44a5dceSAnika Henke    /**
1313b44a5dceSAnika Henke     * Embed audio in HTML
1314b44a5dceSAnika Henke     *
1315b44a5dceSAnika Henke     * @author Anika Henke <anika@selfthinker.org>
1316b44a5dceSAnika Henke     *
1317b44a5dceSAnika Henke     * @param string $src      - ID of audio to embed
13186d4af72aSAnika Henke     * @param array $atts      - additional attributes for the <audio> tag
1319f50634f0SAnika Henke     * @return string
1320b44a5dceSAnika Henke     */
1321b44a5dceSAnika Henke    function _audio($src,$atts=null){
1322b44a5dceSAnika Henke
1323b44a5dceSAnika Henke        // prepare alternative formats
1324b44a5dceSAnika Henke        $extensions = array('ogg', 'mp3', 'wav');
1325b44a5dceSAnika Henke        $alternatives = media_alternativefiles($src, $extensions);
1326b44a5dceSAnika Henke
1327f50634f0SAnika Henke        $out = '';
1328b44a5dceSAnika Henke        // open audio tag
1329f50634f0SAnika Henke        $out .= '<audio '.buildAttributes($atts).' controls="controls">'.NL;
13303641199aSAnika Henke        $fallback = '';
1331b44a5dceSAnika Henke
1332b44a5dceSAnika Henke        // output source for each alternative audio format
1333b44a5dceSAnika Henke        foreach($alternatives as $mime => $file) {
1334b44a5dceSAnika Henke            $url = ml($file,array('cache'=>$cache),true,'&');
133517954bb5SAnika Henke            $title = $atts['title'] ? $atts['title'] : $this->_xmlEntities(utf8_basename(noNS($file)));
1336b44a5dceSAnika Henke
1337f50634f0SAnika Henke            $out .= '<source src="'.hsc($url).'" type="'.$mime.'" />'.NL;
1338b44a5dceSAnika Henke            // alternative content (just a link to the file)
13393641199aSAnika Henke            $fallback .= $this->internalmedia($file, $title, NULL, NULL, NULL, $cache=NULL, $linking='linkonly', $return=true);
1340b44a5dceSAnika Henke        }
1341b44a5dceSAnika Henke
1342b44a5dceSAnika Henke        // finish
13433641199aSAnika Henke        $out .= $fallback;
1344f50634f0SAnika Henke        $out .= '</audio>'.NL;
1345f50634f0SAnika Henke        return $out;
1346b44a5dceSAnika Henke    }
1347b44a5dceSAnika Henke
13480cecf9d5Sandi}
13490cecf9d5Sandi
1350e3776c06SMichael Hamann//Setup VIM: ex: et ts=4 :
1351