xref: /plugin/ditaa/syntax.php (revision 95615d6c9d0f4f656d9f64521d7ea9c14a40d45e)
1a34ed36bSDennis Ploeger<?php
2a34ed36bSDennis Ploeger/**
3a34ed36bSDennis Ploeger * Ditaa-Plugin: Converts Ascii-Flowcharts into a png-File
4a34ed36bSDennis Ploeger *
5a34ed36bSDennis Ploeger * @license     GPL 2 (http://www.gnu.org/licenses/gpl.html)
6a34ed36bSDennis Ploeger * @author      Dennis Ploeger <develop [at] dieploegers [dot] de>
7a34ed36bSDennis Ploeger * @author      Christoph Mertins <c [dot] mertins [at] gmail [dot] com>
8d402f512SAndreas Gohr * @author      Andreas Gohr <andi@splitbrain.org>
9a34ed36bSDennis Ploeger */
10a34ed36bSDennis Ploeger
11a34ed36bSDennis Ploegerif(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../../').'/');
12a34ed36bSDennis Ploegerif(!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/');
13a34ed36bSDennis Ploegerrequire_once(DOKU_PLUGIN.'syntax.php');
14a34ed36bSDennis Ploeger
15a34ed36bSDennis Ploegerclass syntax_plugin_ditaa extends DokuWiki_Syntax_Plugin {
16a34ed36bSDennis Ploeger
17a34ed36bSDennis Ploeger    /**
18a34ed36bSDennis Ploeger     * What about paragraphs?
19a34ed36bSDennis Ploeger     */
20a34ed36bSDennis Ploeger    function getPType(){
21a34ed36bSDennis Ploeger        return 'normal';
22a34ed36bSDennis Ploeger    }
23a34ed36bSDennis Ploeger
24a34ed36bSDennis Ploeger    /**
25a34ed36bSDennis Ploeger     * What kind of syntax are we?
26a34ed36bSDennis Ploeger     */
27a34ed36bSDennis Ploeger    function getType(){
28a34ed36bSDennis Ploeger        return 'substition';
29a34ed36bSDennis Ploeger    }
30a34ed36bSDennis Ploeger
31a34ed36bSDennis Ploeger    /**
32a34ed36bSDennis Ploeger     * Where to sort in?
33a34ed36bSDennis Ploeger     */
34a34ed36bSDennis Ploeger    function getSort(){
35a34ed36bSDennis Ploeger        return 200;
36a34ed36bSDennis Ploeger    }
37a34ed36bSDennis Ploeger
38a34ed36bSDennis Ploeger    /**
3972a2bf1dSAndreas Gohr     * Connect pattern to lexer
40a34ed36bSDennis Ploeger     */
41a34ed36bSDennis Ploeger
42a34ed36bSDennis Ploeger    function connectTo($mode) {
4372a2bf1dSAndreas Gohr        $this->Lexer->addSpecialPattern('<ditaa.*?>\n.*?\n</ditaa>',$mode,'plugin_ditaa');
44a34ed36bSDennis Ploeger    }
45a34ed36bSDennis Ploeger
46a34ed36bSDennis Ploeger    /**
47a34ed36bSDennis Ploeger     * Handle the match
48a34ed36bSDennis Ploeger     */
4972a2bf1dSAndreas Gohr    function handle($match, $state, $pos, &$handler) {
502a956e66SAndreas Gohr        $info = $this->getInfo();
512a956e66SAndreas Gohr
5272a2bf1dSAndreas Gohr        // prepare default data
5372a2bf1dSAndreas Gohr        $return = array(
5472a2bf1dSAndreas Gohr                        'width'     => 0,
5572a2bf1dSAndreas Gohr                        'height'    => 0,
5672a2bf1dSAndreas Gohr                        'antialias' => true,
5772a2bf1dSAndreas Gohr                        'edgesep'   => true,
5872a2bf1dSAndreas Gohr                        'round'     => false,
5972a2bf1dSAndreas Gohr                        'shadow'    => true,
6072a2bf1dSAndreas Gohr                        'scale'     => 1,
6172a2bf1dSAndreas Gohr                        'align'     => '',
622a956e66SAndreas Gohr                        'version'   => $info['date'], //forece rebuild of images on update
6372a2bf1dSAndreas Gohr                       );
64a34ed36bSDennis Ploeger
65a34ed36bSDennis Ploeger
6672a2bf1dSAndreas Gohr        // prepare input
6772a2bf1dSAndreas Gohr        $lines = explode("\n",$match);
6872a2bf1dSAndreas Gohr        $conf = array_shift($lines);
6972a2bf1dSAndreas Gohr        array_pop($lines);
7072a2bf1dSAndreas Gohr
7172a2bf1dSAndreas Gohr        // match config options
7272a2bf1dSAndreas Gohr        if(preg_match('/\b(left|center|right)\b/i',$conf,$match)) $return['align'] = $match[1];
7372a2bf1dSAndreas Gohr        if(preg_match('/\b(\d+)x(\d+)\b/',$conf,$match)){
7472a2bf1dSAndreas Gohr            $return['width']  = $match[1];
7572a2bf1dSAndreas Gohr            $return['height'] = $match[2];
76a34ed36bSDennis Ploeger        }
772a956e66SAndreas Gohr        if(preg_match('/\b(\d+(\.\d+)?)X\b/',$conf,$match)) $return['scale']  = $match[1];
7872a2bf1dSAndreas Gohr        if(preg_match('/\bwidth=([0-9]+)\b/i', $conf,$match)) $return['width'] = $match[1];
7972a2bf1dSAndreas Gohr        if(preg_match('/\bheight=([0-9]+)\b/i', $conf,$match)) $return['height'] = $match[1];
8072a2bf1dSAndreas Gohr        // match boolean toggles
8172a2bf1dSAndreas Gohr        if(preg_match_all('/\b(no)?(antialias|edgesep|round|shadow)\b/i',$conf,$matches,PREG_SET_ORDER)){
8272a2bf1dSAndreas Gohr            foreach($matches as $match){
8372a2bf1dSAndreas Gohr                $return[$match[2]] = ! $match[1];
8472a2bf1dSAndreas Gohr            }
8572a2bf1dSAndreas Gohr        }
8672a2bf1dSAndreas Gohr
8705cbae88SAndreas Gohr        $input = join("\n",$lines);
8805cbae88SAndreas Gohr        $return['md5'] = md5($input); // we only pass a hash around
8905cbae88SAndreas Gohr
9005cbae88SAndreas Gohr        // store input for later use
9105cbae88SAndreas Gohr        io_saveFile($this->_cachename($return,'txt'),$input);
9272a2bf1dSAndreas Gohr
9372a2bf1dSAndreas Gohr        return $return;
94a34ed36bSDennis Ploeger    }
95a34ed36bSDennis Ploeger
96a34ed36bSDennis Ploeger    /**
97*95615d6cSAndreas Gohr     * Cache file is based on parameters that influence the result image
9805cbae88SAndreas Gohr     */
9905cbae88SAndreas Gohr    function _cachename($data,$ext){
10005cbae88SAndreas Gohr        unset($data['width']);
10105cbae88SAndreas Gohr        unset($data['height']);
10205cbae88SAndreas Gohr        unset($data['align']);
10305cbae88SAndreas Gohr        return getcachename(join('x',array_values($data)),'.ditaa.'.$ext);
10405cbae88SAndreas Gohr    }
10505cbae88SAndreas Gohr
10605cbae88SAndreas Gohr    /**
107*95615d6cSAndreas Gohr     * Create output
108*95615d6cSAndreas Gohr     */
109*95615d6cSAndreas Gohr    function render($format, &$R, $data) {
110*95615d6cSAndreas Gohr        if($format == 'xhtml'){
111*95615d6cSAndreas Gohr            $img = DOKU_BASE.'lib/plugins/ditaa/img.php?'.buildURLparams($data);
112*95615d6cSAndreas Gohr            $R->doc .= '<img src="'.$img.'" class="media'.$data['align'].'" alt=""';
113*95615d6cSAndreas Gohr            if($data['width'])  $R->doc .= ' width="'.$data['width'].'"';
114*95615d6cSAndreas Gohr            if($data['height']) $R->doc .= ' height="'.$data['height'].'"';
115*95615d6cSAndreas Gohr            if($data['align'] == 'right') $ret .= ' align="right"';
116*95615d6cSAndreas Gohr            if($data['align'] == 'left')  $ret .= ' align="left"';
117*95615d6cSAndreas Gohr            $R->doc .= '/>';
118*95615d6cSAndreas Gohr            return true;
119*95615d6cSAndreas Gohr        }elseif($format == 'odt'){
120*95615d6cSAndreas Gohr            $src = $this->_imgfile($data);
121*95615d6cSAndreas Gohr            $R->_odtAddImage($src,$data['width'],$data['height'],$data['align']);
122*95615d6cSAndreas Gohr            return true;
123*95615d6cSAndreas Gohr        }
124*95615d6cSAndreas Gohr        return false;
125*95615d6cSAndreas Gohr    }
126*95615d6cSAndreas Gohr
127*95615d6cSAndreas Gohr
128*95615d6cSAndreas Gohr    /**
129*95615d6cSAndreas Gohr     * Return path to the rendered image on our local system
130*95615d6cSAndreas Gohr     */
131*95615d6cSAndreas Gohr    function _imgfile($data){
132*95615d6cSAndreas Gohr        $cache  = $this->_cachename($data,'png');
133*95615d6cSAndreas Gohr
134*95615d6cSAndreas Gohr        // create the file if needed
135*95615d6cSAndreas Gohr        if(!file_exists($cache)){
136*95615d6cSAndreas Gohr            $in = $this->_cachename($data,'txt');
137*95615d6cSAndreas Gohr            if($this->getConf('java')){
138*95615d6cSAndreas Gohr                $ok = $this->_run($data,$in,$cache);
139*95615d6cSAndreas Gohr            }else{
140*95615d6cSAndreas Gohr                $ok = $this->_remote($data,$in,$cache);
141*95615d6cSAndreas Gohr            }
142*95615d6cSAndreas Gohr            if(!$ok) return false;
143*95615d6cSAndreas Gohr            clearstatcache();
144*95615d6cSAndreas Gohr        }
145*95615d6cSAndreas Gohr
146*95615d6cSAndreas Gohr        // resized version
147*95615d6cSAndreas Gohr        if($data['width']){
148*95615d6cSAndreas Gohr            $cache = media_resize_image($cache,'png',$data['width'],$data['height']);
149*95615d6cSAndreas Gohr        }
150*95615d6cSAndreas Gohr
151*95615d6cSAndreas Gohr        // something went wrong, we're missing the file
152*95615d6cSAndreas Gohr        if(!file_exists($cache)) return false;
153*95615d6cSAndreas Gohr
154*95615d6cSAndreas Gohr        return $cache;
155*95615d6cSAndreas Gohr    }
156*95615d6cSAndreas Gohr
157*95615d6cSAndreas Gohr    /**
15805cbae88SAndreas Gohr     * Render the output remotely at ditaa.org
15905cbae88SAndreas Gohr     */
16005cbae88SAndreas Gohr    function _remote($data,$in,$out){
161*95615d6cSAndreas Gohr        if(!file_exists($in)){
162*95615d6cSAndreas Gohr            if($conf['debug']){
163*95615d6cSAndreas Gohr                dbglog($in,'no such ditaa input file');
164*95615d6cSAndreas Gohr            }
165*95615d6cSAndreas Gohr            return false;
166*95615d6cSAndreas Gohr        }
167*95615d6cSAndreas Gohr
16805cbae88SAndreas Gohr        $http = new DokuHTTPClient();
16905cbae88SAndreas Gohr        $http->timeout=30;
17005cbae88SAndreas Gohr
17105cbae88SAndreas Gohr        $pass = array();
17205cbae88SAndreas Gohr        $pass['scale']   = $data['scale'];
17305cbae88SAndreas Gohr        $pass['timeout'] = 25;
17405cbae88SAndreas Gohr        $pass['grid']    = io_readFile($in);
17505cbae88SAndreas Gohr        if(!$data['antialias']) $pass['A'] = 'on';
17605cbae88SAndreas Gohr        if(!$data['shadow'])    $pass['S'] = 'on';
17705cbae88SAndreas Gohr        if($data['round'])      $pass['r'] = 'on';
17805cbae88SAndreas Gohr        if(!$data['edgesep'])   $pass['E'] = 'on';
17905cbae88SAndreas Gohr
18005cbae88SAndreas Gohr        $img = $http->post('http://ditaa.org/ditaa/render',$pass);
18105cbae88SAndreas Gohr        if(!$img) return false;
18205cbae88SAndreas Gohr
183*95615d6cSAndreas Gohr        return io_saveFile($out,$img);
18405cbae88SAndreas Gohr    }
18505cbae88SAndreas Gohr
186a34ed36bSDennis Ploeger    /**
1872a956e66SAndreas Gohr     * Run the ditaa Java program
188a34ed36bSDennis Ploeger     */
18905cbae88SAndreas Gohr    function _run($data,$in,$out) {
1902a956e66SAndreas Gohr        global $conf;
191a34ed36bSDennis Ploeger
19205cbae88SAndreas Gohr        if(!file_exists($in)){
19305cbae88SAndreas Gohr            if($conf['debug']){
19405cbae88SAndreas Gohr                dbglog($in,'no such ditaa input file');
19505cbae88SAndreas Gohr            }
19605cbae88SAndreas Gohr            return false;
19705cbae88SAndreas Gohr        }
198a34ed36bSDennis Ploeger
1992a956e66SAndreas Gohr        $cmd  = $this->getConf('java');
20036721e14SAndreas Gohr        $cmd .= ' -Djava.awt.headless=true -Dfile.encoding=UTF-8 -jar';
2012a956e66SAndreas Gohr        $cmd .= ' '.escapeshellarg(dirname(__FILE__).'/ditaa/ditaa0_9.jar'); //ditaa jar
20236721e14SAndreas Gohr        $cmd .= ' --encoding UTF-8';
20305cbae88SAndreas Gohr        $cmd .= ' '.escapeshellarg($in); //input
20405cbae88SAndreas Gohr        $cmd .= ' '.escapeshellarg($out); //output
2052a956e66SAndreas Gohr        $cmd .= ' -s '.escapeshellarg($data['scale']);
2062a956e66SAndreas Gohr        if(!$data['antialias']) $cmd .= ' -A';
2072a956e66SAndreas Gohr        if(!$data['shadow'])    $cmd .= ' -S';
2082a956e66SAndreas Gohr        if($data['round'])      $cmd .= ' -r';
2092a956e66SAndreas Gohr        if(!$data['edgesep'])   $cmd .= ' -E';
210a34ed36bSDennis Ploeger
211a34ed36bSDennis Ploeger        exec($cmd, $output, $error);
212a34ed36bSDennis Ploeger
2133c74ed32SAndreas Gohr        if ($error != 0){
2143c74ed32SAndreas Gohr            if($conf['debug']){
2153c74ed32SAndreas Gohr                dbglog(join("\n",$output),'ditaa command failed: '.$cmd);
2163c74ed32SAndreas Gohr            }
2173c74ed32SAndreas Gohr            return false;
2183c74ed32SAndreas Gohr        }
21905cbae88SAndreas Gohr
220a34ed36bSDennis Ploeger        return true;
221a34ed36bSDennis Ploeger    }
222a34ed36bSDennis Ploeger
223a34ed36bSDennis Ploeger}
224a34ed36bSDennis Ploeger
225