4 * Copyright 2004 $ThePhpWikiProgrammingTeam
6 * This file is part of PhpWiki.
8 * PhpWiki is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * PhpWiki is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License along
19 * with PhpWiki; if not, write to the Free Software Foundation, Inc.,
20 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
24 * The Ploticus plugin passes all its arguments to the ploticus
25 * binary and displays the result as PNG, GIF, EPS, SVG or SWF.
26 * Ploticus is a free, GPL, non-interactive software package
27 * for producing plots, charts, and graphics from data.
28 * See http://ploticus.sourceforge.net/doc/welcome.html
30 * @Author: Reini Urban
33 * - For windows you need either a gd library with GIF support or
34 * a ploticus with PNG support. This comes e.g. with the cygwin build.
35 * - We support only images supported by GD so far (PNG most likely).
36 * No EPS, PS, SWF, SVG or SVGZ support yet, due to limitations in WikiPluginCached.
37 * This will be fixed soon.
40 <<Ploticus device=png [ploticus options...]
41 multiline ploticus script ...
43 * or without any script: (not tested)
44 <<Ploticus -prefab vbars data=myfile.dat delim=tab y=1 clickmapurl="http://mywiki.url/wiki/?pagename=@2" clickmaplabel="@3" -csmap >>
46 * TODO: PloticusSql - create intermediate data from SQL. Similar to SqlResult, just in graphic form.
47 * For example to produce nice looking pagehit statistics or ratings statistics.
48 * Ploticus has its own sql support within #getproc data, but this would expose security information.
51 if (!defined("PLOTICUS_EXE"))
53 define('PLOTICUS_EXE', 'pl.exe');
55 define('PLOTICUS_EXE', '/usr/local/bin/pl');
56 //TODO: check $_ENV['PLOTICUS_PREFABS'] and default directory
58 require_once 'lib/WikiPluginCached.php';
60 class WikiPlugin_Ploticus
61 extends WikiPluginCached
64 * Sets plugin type to MAP if -csmap (-map or -mapdemo or -csmapdemo not supported)
65 * or HTML if the imagetype is not supported by GD (EPS, SVG, SVGZ) (not yet)
66 * or IMG_INLINE if device = png, gif or jpeg
68 function getPluginType()
70 if (!empty($this->_args['-csmap']))
71 return PLUGIN_CACHED_MAP; // not yet tested
72 // produce these on-demand so far, uncached.
73 // will get better support in WikiPluginCached soon.
74 // FIXME: html also? what about ''?
75 $type = $this->decideImgType($this->_args['device']);
76 if ($type == $this->_args['device'])
77 return PLUGIN_CACHED_IMG_INLINE;
78 $device = strtolower($this->_args['device']);
79 if (in_array($device, array('svg', 'swf', 'svgz', 'eps', 'ps', 'pdf', 'html'))) {
80 switch ($this->_args['device']) {
83 return PLUGIN_CACHED_STATIC | PLUGIN_CACHED_SVG_PNG;
85 return PLUGIN_CACHED_STATIC | PLUGIN_CACHED_SWF;
87 return PLUGIN_CACHED_STATIC | PLUGIN_CACHED_HTML;
90 return PLUGIN_CACHED_IMG_INLINE; // normal cached libgd image handles
98 function getDescription()
100 return _("Ploticus image creation");
103 function managesValidators()
108 function getDefaultArguments()
111 'device' => 'png', // png,gif,svgz,svg,...
114 'data' => false, // <!plugin-list !> support
120 function handle_plugin_args_cruft(&$argstr, &$args)
122 $this->source = $argstr;
126 * Sets the expire time to one day (so the image producing
127 * functions are called seldomly) or to about two minutes
128 * if a help screen is created.
130 function getExpire($dbi, $argarray, $request)
132 if (!empty($argarray['help']))
133 return '+120'; // 2 minutes
134 return sprintf('+%d', 3 * 86000); // approx 3 days
138 * Sets the imagetype according to user wishes and
139 * relies on WikiPluginCached to catch illegal image
141 * (I feel unsure whether this option is reasonable in
142 * this case, because png will definitely have the
146 * @param array $argarray
147 * @param Request $request
148 * @return string 'png', 'jpeg', 'gif'
150 function getImageType($dbi, $argarray, $request)
152 return $argarray['device'];
156 * This gives an alternative text description of
159 function getAlt($dbi, $argstr, $request)
161 return (!empty($this->_args['alt'])) ? $this->_args['alt']
162 : $this->getDescription();
166 * Returns an image containing a usage description of the plugin.
168 * TODO: -csmap pointing to the Ploticus documentation at sf.net.
169 * @return string image handle
173 $def = $this->defaultArguments();
174 //$other_imgtypes = $GLOBALS['PLUGIN_CACHED_IMGTYPES'];
175 //unset ($other_imgtypes[$def['imgtype']]);
178 'device' => ' = "' . $def['device'] . "(default)|"
179 . join('|', $GLOBALS['PLUGIN_CACHED_IMGTYPES']) . '"',
180 'data' => ' <!plugin-list !>: pagelist as input',
181 'alt' => ' = "alternate text"',
182 '-csmap' => ' bool: clickable map?',
183 'help' => ' bool: displays this screen',
184 '...' => ' all further lines below the first plugin line ',
185 '' => ' and inside the tags are the ploticus script.',
189 foreach ($helparr as $alignright => $alignleft) {
190 $length = max($length, strlen($alignright));
193 foreach ($helparr as $alignright => $alignleft) {
194 $helptext .= substr(' '
195 . $alignright, -$length) . $alignleft . "\n";
197 return $this->text2img($helptext, 4, array(1, 0, 0),
198 array(255, 255, 255));
201 function withShellCommand($script)
204 $pos = strpos($script, $findme); // uppercase?
210 function getImage($dbi, $argarray, $request)
212 //extract($this->getArgs($argstr, $request));
213 //extract($argarray);
214 $source =& $this->source;
215 if (!empty($source)) {
216 if ($this->withShellCommand($source)) {
217 $this->_errortext .= _("shell commands not allowed in Ploticus");
220 if (is_array($argarray['data'])) { // support <!plugin-list !> pagelists
221 $src = "#proc getdata\ndata:";
223 foreach ($argarray['data'] as $data) {
226 $src .= ("\t" . join(" ", $data) . "\n");
228 $src .= ("\t" . '"' . $data . '" ' . $i++ . "\n");
233 $tempfile = $this->tempnam('Ploticus', 'plo');
235 $gif = $argarray['device'];
236 $args = "-$gif -o $tempfile.$gif";
237 if (!empty($argarray['-csmap'])) {
238 $args .= " -csmap -mapfile $tempfile.map";
239 $this->_mapfile = "$tempfile.map";
241 if (!empty($argarray['-prefab'])) {
242 //check $_ENV['PLOTICUS_PREFABS'] and default directory
243 global $HTTP_ENV_VARS;
244 if (empty($HTTP_ENV_VARS['PLOTICUS_PREFABS'])) {
245 if (file_exists("/usr/share/ploticus"))
246 $HTTP_ENV_VARS['PLOTICUS_PREFABS'] = "/usr/share/ploticus";
247 elseif (defined('PLOTICUS_PREFABS'))
248 $HTTP_ENV_VARS['PLOTICUS_PREFABS'] = constant('PLOTICUS_PREFABS');
250 $args .= (" -prefab " . $argarray['-prefab']);
253 $fp = fopen("$tempfile.plo", "w");
254 fwrite($fp, $source);
256 $code = $this->execute(PLOTICUS_EXE . " $tempfile.plo $args", $tempfile . ".$gif");
258 $code = $this->filterThroughCmd($source, PLOTICUS_EXE . " -stdin $args");
262 // return $this->error(fmt("Couldn't start commandline '%s'", $commandLine));
263 if (!file_exists($tempfile . ".$gif")) {
264 $this->_errortext .= sprintf(_("%s error: outputfile '%s' not created"),
265 "Ploticus", "$tempfile.$gif");
267 $this->_errortext .= ("\ncmd-line: " . PLOTICUS_EXE . " $tempfile.plo $args");
269 $this->_errortext .= ("\ncmd-line: cat script | " . PLOTICUS_EXE . " $args");
270 @unlink("$tempfile.pl");
271 @unlink("$tempfile");
274 $ImageCreateFromFunc = "ImageCreateFrom$gif";
275 if (function_exists($ImageCreateFromFunc)) {
276 $handle = $ImageCreateFromFunc("$tempfile.$gif");
278 @unlink("$tempfile.$gif");
279 @unlink("$tempfile.plo");
280 @unlink("$tempfile");
284 return "$tempfile.$gif";
286 return $this->error(fmt("empty source"));
290 // which argument must be set to 'png', for the fallback image when svg will fail on the client.
297 function getMap($dbi, $argarray, $request)
299 $img = $this->getImage($dbi, $argarray, $request);
300 return array($this->_mapfile, $img);
308 // c-hanging-comment-ender-p: nil
309 // indent-tabs-mode: nil