3 #define("FONT_TAHOMA", 'fonts/tahoma.ttf');
4 #define("FONT_DEJAVUSANS", 'fonts/DejaVuSans.ttf');
5 #define("FONT_DEJAVUSANS_CONDENSED", 'fonts/DejaVuSansCondensed.ttf');
6 #define("FONT_DEJAVUSERIF", 'fonts/DejaVuSerif.ttf');
7 #define("FONT_DEJAVUSERIF_CONDENSED", 'fonts/DejaVuSerifCondensed.ttf');
9 #define("FONT_TAHOMA", Kohana::find_file('fonts', 'tahoma', 'ttf'));
10 #define("FONT_DEJAVUSANS", Kohana::find_file('fonts', 'DejaVuSans', 'ttf'));
11 #define("FONT_DEJAVUSANS_CONDENSED", Kohana::find_file('vendor', 'mfchart/fonts/DejaVuSansCondensed', false, 'ttf'));
12 #define("FONT_DEJAVUSERIF", Kohana::find_file('fonts', 'DejaVuSerif', 'ttf'));
13 #define("FONT_DEJAVUSERIF_CONDENSED", Kohana::find_file('fonts', 'DejaVuSerifCondensed', 'ttf'));
18 protected $image; // image with the graph
20 protected $width = 450; // width of the graph
22 protected $height = 180; // height of the graph
24 protected $margin_left = 0; // margin around the graph where the legend is displayed
26 protected $margin_bottom = 0; // margin around the graph where the legend is displayed
28 protected $margin_right = 0; // margin around the graph where the legend is displayed
30 protected $margin_top = 0; // margin around the graph where the legend is displayed
32 protected $graph_width; // graph plot area
34 protected $graph_height; // graph plot area
36 protected $values = array(); // values for displaying in the graph
38 protected $font = FONT_DEJAVUSANS_CONDENSED
; // hardcoded for now
40 protected $font_size = 8; // GD2 - points (but GD1 - pixels)
42 protected $colors = array(); // for colors init -- items are arrays in format array(hex_color, alpha, allocated object) or array(array(hex_color, alpha, allocated object), ...)
44 protected $title = '';
46 protected $legend = array(); // legend for the graph
48 protected $occupied_areas = array(); // for collision detect when placing some items like legend box or labels for points in line graphs etc.
49 // each item in format array(x1, y1, x2, y2)
51 public function __construct($width=NULL, $height=NULL)
54 $this->width
= (int) $width;
56 $this->height
= (int) $height;
58 $this->colors
['background_color'] = array('#fdfdfd', NULL, NULL); // background of the generated image
59 #$this->colors['background_color'] = array('#f6f7f8', NULL, NULL); // background of the generated image
60 $this->colors
['border_color'] = array('#fdfdfd', NULL, NULL); // border of the generated image
61 #$this->colors['border_color'] = array(NULL, NULL, NULL); // border of the generated image
62 $this->colors
['font_color'] = array('#595959', NULL, NULL); // values at axis
63 $this->colors
['font_color2'] = array('#000000', NULL, NULL); // legend at axis & legend
64 $this->colors
['font_color3'] = array('#d08a22', NULL, NULL); // values at bars
65 $this->colors
['legend_color'] = array('#fefefe', NULL, NULL); // background legend color
66 $this->colors
['shade_color'] = array('#666666', 95, NULL);
67 $this->colors
['shade_color2'] = array('#ffffff', NULL, NULL);
68 $this->colors
['shade_color3'] = array('#ffffff', NULL, NULL);
69 # $this->colors['shade_color2'] = array('#e6e6e6', NULL, NULL);
70 # $this->colors['shade_color3'] = array('#f0f0f0', NULL, NULL);
74 * You can set one margin for all by setting only firs parameter.
75 * Or you can set vertical and horizontal margin by setting two parameters.
78 * @param int left margin
79 * @param int top margin
80 * @param int right margin
81 * @param int bottom margin
83 public function set_margins($left, $top=NULL, $right=NULL, $bottom=NULL)
85 $this->margin_left
= $left;
86 $this->margin_top
= ($top === NULL) ?
$this->margin_left
: $top;
87 $this->margin_right
= ($right === NULL) ?
$this->margin_left
: $right;
88 $this->margin_bottom
= ($bottom === NULL) ?
$this->margin_top
: $bottom;
91 public function set_font($font)
96 public function set_font_size($size)
98 $this->font_size
= $size;
101 public function set_legend($values)
103 $this->legend
= $values;
106 public function set_color($color_name, $color_value, $alpha=NULL)
108 $this->colors
[$color][0] = $color_value;
110 $this->colors
[$color][1] = $alpha;
113 public function set_title($value)
115 $this->title
= $value;
118 // set the values for the graph
119 public function set_data($data, $type=false)
124 // to float - "check"
125 foreach ($data as $key => $row)
127 $key = preg_replace("~[^a-zA-Z0-9 -_]~", null, $key);
133 foreach ($row as $k => $v)
134 $data[$key][$k] = (float) $v;
138 $data[$key] = (float) $row;
142 if ($type === 'pie') {
143 unset($this->colors
['colors']);
144 $this->colors
['colors'] = reports
::get_color_values($labels);
147 $this->values
= $data;
150 // draw image with the graph
151 public function draw()
153 $this->image
= imagecreatetruecolor($this->width
, $this->height
);
154 imagealphablending($this->image
, true);
156 $this->init_colors();
158 // draw background + border
159 if ($this->get_color('border_color'))
161 imagefilledrectangle($this->image
, 0, 0, $this->width
, $this->height
, $this->get_color('border_color')); // draw border
162 imagefilledrectangle($this->image
, 1, 1, $this->width
-2, $this->height
-2, $this->get_color('background_color')); // draw background
166 imagefilledrectangle($this->image
, 0, 0, $this->width
, $this->height
, $this->get_color('background_color')); // draw background
170 if (!empty($this->title
))
172 $box_points = imagettfbbox($this->font_size
, 0, $this->font
, $this->title
);
173 $textheight = $box_points[3]-$box_points[5];
174 $textwidth = $box_points[4]-$box_points[6];
175 utilities
::imagestringbox($this->image
, $this->font
, $this->font_size
, 0, $this->margin_top
, $this->width
, $this->margin_top+
$textheight, ALIGN_CENTER
, VALIGN_MIDDLE
, 0, $this->title
, $this->get_color('font_color2'));
176 $this->margin_top +
= $textheight +
10;
178 $this->add_occupied($this->width
/2-$textwidth/2, $this->margin_top
, $this->width
/2+
$textwidth/2, $this->margin_top+
$textheight);
182 // get the soruce of the image
183 public function get_image($type = 'png')
189 imagepng($this->image
);
194 imagejpeg($this->image
, '', 0.7);
198 imagegif($this->image
);
202 imagewbmp($this->image
);
205 $img = ob_get_contents();
212 public function display()
214 if (function_exists("imagepng"))
216 header("Content-type: image/png");
217 echo $this->get_image('png');
219 elseif (function_exists("imagejpeg"))
221 header("Content-type: image/jpeg");
222 echo $this->get_image('jpg');
224 elseif (function_exists("imagegif"))
226 header("Content-type: image/gif");
227 echo $this->get_image('gif');
229 elseif (function_exists("imagewbmp"))
231 header("Content-type: image/vnd.wap.wbmp");
232 echo $this->get_image('wbmp');
236 throw new Exception("Doh! No graphical functions on this server?");
242 public function set_width($w)
247 public function set_height($h)
253 // returns size of the graph plot area
254 protected function get_plot_area()
257 $this->width
- $this->margin_left
- $this->margin_right
, // width
258 $this->height
- $this->margin_top
- $this->margin_bottom
// height
262 protected function get_color($key, $i=0, $j=2)
264 if (!isset($this->colors
[$key]))
267 $i = $i %
count($this->colors
[$key]); // if there is not enough colors get it in cycle
269 if ($this->colors
[$key][$i][$j] === NULL)
272 return $this->colors
[$key][$i][$j];
277 private function init_colors()
279 foreach ($this->colors
as $key => $values)
281 if (!is_array($values[0]))
282 $this->colors
[$key] = $values = array($values);
284 foreach ($values as $i => $value)
286 if ($value[0] === NULL)
289 $rgb = utilities
::hex2rgb($value[0]);
291 if ($value[1] === NULL)
292 $this->colors
[$key][$i][2] = imagecolorallocate($this->image
, $rgb[0], $rgb[1], $rgb[2]);
295 $this->colors
[$key][$i][2] = imagecolorallocatealpha($this->image
, $rgb[0], $rgb[1], $rgb[2], $value[1]);
301 * Draw the legend box.
302 * If position of the box isn't set manually it's trying to determine position automatically according to already occupied areas.
304 * @param str Color of the box background.
305 * @param int Relative size of font in points. The font size. Depending on your version of GD, this should be specified as the pixel size (GD1) or point size (GD2).
306 * @param int Position x of the top left of the box.
307 * @param int Position y of the top left of the box.
309 protected function draw_legend($color_index, $relative_font_size=-1, $position_x=NULL, $position_y=NULL)
311 if (empty($this->legend
))
314 $font_legend = ($relative_font_size<0 AND $this->font_size
<abs($relative_font_size)) ?
$this->font_size
: $this->font_size+
$relative_font_size;
318 foreach ($this->legend
as $l)
320 $box_points = imagettfbbox($font_legend, 0, $this->font
, $l);
321 $width = $box_points[4]-$box_points[6];
322 $height = $box_points[3]-$box_points[5];
323 if ($maxwidth < $width)
325 if ($maxheight < $height)
326 $maxheight = $height;
328 $maxheight +
= $maxheight*0.3; // line spacing
330 $border = $maxheight*0.5;
332 if ($position_x === NULL AND $position_y === NULL)
334 $from = array($this->margin_left+
5, $this->margin_top
-10 > 0 ?
$this->margin_top
-10 : 0);
335 $to = array($this->width
, $this->height
-$this->margin_bottom
);
336 $found = $this->place_in_free($maxwidth+
$border*2+
10, count($this->legend
)*$maxheight+
$border*2, 5, $from, $to);
337 $position_x = $found[0];
338 $position_y = $found[1];
341 $legend_x1 = $position_x;
342 $legend_x2 = $position_x +
$maxwidth +
$border*2 +
10;
343 $legend_y1 = $position_y;
344 $legend_y2 = $position_y +
count($this->legend
)*$maxheight +
$border*2;
346 // $legend_x1 = $this->width-$maxwidth-40;
347 // $legend_x2 = $legend_x1 + $maxwidth + 30;
348 // $legend_y1 = $this->margin_top-10;
349 // $legend_y2 = $legend_y1 + count($this->legend)*$maxheight + 10;
351 utilities
::imagefillroundedrect($this->image
, $legend_x1+
2, $legend_y1+
2, $legend_x2+
2, $legend_y2+
2, 5, $this->get_color('shade_color3'));
352 utilities
::imagefillroundedrect($this->image
, $legend_x1+
1, $legend_y1+
1, $legend_x2+
1, $legend_y2+
1, 5, $this->get_color('shade_color2'));
353 utilities
::imagefillroundedrect($this->image
, $legend_x1, $legend_y1, $legend_x2, $legend_y2, 5, $this->get_color('legend_color'));
357 foreach ($this->legend
as $l)
359 $color = $this->get_color($color_index, $i);
360 if( $color !== false ) {
361 $y = $legend_y1 +
5 +
$maxheight*$yi;
362 imagefilledrectangle($this->image
, $legend_x1+
$border, $y+
$maxheight/2-2, $legend_x1+
$border+
5, $y+
$maxheight/2+
3, $color);
363 utilities
::imagestringbox($this->image
, $this->font
, $font_legend, $legend_x1+
$border+
10, $y, $legend_x2, $y+
$maxheight, ALIGN_LEFT
, VALIGN_MIDDLE
, 0, $l, $this->get_color('font_color2'));
370 protected function add_occupied($x1, $y1, $x2, $y2)
374 list($x1, $x2) = array($x2, $x1);
376 list($y1, $y2) = array($y2, $y1);
378 $this->occupied_areas
[] = array(
387 * Detect collision of two boxes. Try to find free place where the box could be placed.
388 * Searching in columns priority now. Returns the solution which is the closest to the top or bottom border in the first column where some solution has beend found.
389 * It's brute-force. I think it could be optimised when some ordering is used for $occupied_areas.
390 * Supports only rectangles aligned with x and y axis.
392 * NOTE: For use also with rotated rectangles (for lines for example) look at "separating axis test" at Internet and implement it :)
393 * http://en.wikipedia.org/wiki/Separating_axis_theorem
394 * http://board.flashkit.com/board/showthread.php?t=787281
396 * @param int Width of the box.
397 * @param int Height of the box.
398 * @param int Padding from the collision boxes. Working correctly only for the last found collision now :((
399 * @param array Where should the searching start? Point [x,y].
400 * @param array Where should the searching end? Point [x,y].
401 * @param [left,right] Direction of searching in x axis.
402 * @param [up,down] Direction of searching in y axis.
404 protected function place_in_free($width, $height, $padding=5, $from=array(0,0), $to=array(NULL,NULL), $direction_x = 'left', $direction_y = 'down')
406 $width +
= $padding*2; // consider padding
407 $height +
= $padding*2; // consider padding
409 // if it's not specified set it to the right bottom corner of the image
411 $to[0] = $this->width
;
413 $to[1] = $this->height
;
415 // order from and to points
416 if ($from[0] > $to[0])
417 list($from[0], $to[0]) = array($to[0], $from[0]);
418 if ($from[1] > $to[1])
419 list($from[1], $to[1]) = array($to[1], $from[1]);
422 if ($direction_x == 'left')
432 $x2 = $from[0]+
$width;
434 if ($direction_y == 'up')
437 $y1 = $to[1]-$height;
444 $y2 = $from[1]+
$height;
452 foreach ($this->occupied_areas
as $occupied) // cycle through all occupied areas - brute-force
454 if ($x2>=$occupied[0] AND $x1<=$occupied[2] AND $y1<=$occupied[3] AND $y2>=$occupied[1])
455 { // collision of boxes
458 // step for axis y movement
459 if ($direction_y == 'down')
460 $sy = ($occupied[3]-$y1+
1)*$step_y;
462 $sy = abs($occupied[1]-$y2+
1)*$step_y;
464 if ($direction_y == 'down' AND $y2+
abs($sy) > $to[1])
465 { // go to the next column
466 if ($found !== NULL) // there is already found solution
469 $y2 = $from[1]+
$height;
473 elseif ($direction_y == 'up' AND $y1-abs($sy) < $from[1])
474 { // go to the next column
475 if ($found !== NULL) // there is already found solution
477 $y1 = $to[1]-$height;
488 if ($x1<$from[0] OR $x2>$to[0]) // we are at the borders (left or right) and nothing found, go out
495 if (!$intersect) // found solution
496 { // store found solution and try to look for better one in this column
499 $found = array($x1, $y1, $x2, $y2);
502 { // replace previous solution only if it's more closely to the border (top or bottom)
503 $gaps = array($found[1]-$from[1], $to[1]-$found[3], $y1-$from[1], $to[1]-$y2);
504 if (($gaps[2]<$gaps[0] AND $gaps[2]<$gaps[1]) OR ($gaps[3]<$gaps[0] AND $gaps[3]<$gaps[1]))
505 $found = array($x1, $y1, $x2, $y2);
508 // simulate intersection
512 if (($direction_y == 'down' AND $y2+
$step_y > $to[1]) OR ($direction_y == 'up' AND $y1-$step_y < $from[1])) // border found
520 if ($found !== NULL) // return found solution
521 list($x1, $y1, $x2, $y2) = $found;
523 return array($x1, $y1, $x2, $y2);
527 // This one is for future purpuses if anything needs to be rewritten. Maybe it would be helpful.
528 // There is approach without switching in the code and also with switching. It's badly mixed...
530 // Switched 'left' and 'right'. Need to be corrected.
532 // protected function place_in_free($width, $height, $padding=5, $from=array(0,0), $to=array(NULL,NULL), $direction_x = 'left', $direction_y = 'down')
534 // $width += $padding*2; // consider padding
535 // $height += $padding*2; // consider padding
537 // // if it's not specified set it to the right bottom corner of the image
538 // if ($to[0] === NULL)
539 // $to[0] = $this->width;
540 // if ($to[1] === NULL)
541 // $to[1] = $this->height;
543 // // order from and to points
544 // if ($from[0] > $to[0])
545 // list($from[0], $to[0]) = array($to[0], $from[0]);
546 // if ($from[1] > $to[1])
547 // list($from[1], $to[1]) = array($to[1], $from[1]);
549 // // init searched box
550 // if ($direction_x == 'right')
553 // $x1 = $to[0]-$width;
560 // $x2 = $from[0]+$width;
562 // if ($direction_y == 'up')
565 // $y1 = $to[1]-$height;
572 // $y2 = $from[1]+$height;
575 // $switch = FALSE; // used for switching between upper and bottom border so it's searching from the borders to the vertical center
577 // $intersect = TRUE;
578 // while ($intersect)
580 // $intersect = FALSE;
581 // foreach ($this->occupied_areas as $occupied) // cycle through all occupied areas - brute-force
583 // if ($x2>=$occupied[0] AND $x1<=$occupied[2] AND $y1<=$occupied[3] AND $y2>=$occupied[1])
584 // { // collision of boxes
585 // $intersect = TRUE;
587 // $sx = ($occupied[2]-$occupied[0])*$step_x; // step for axis x movement
590 // if ($direction_x == 'left' AND $x2+abs($sx) > $to[0])
593 // $x2 = $from[0]+$width;
597 // elseif ($direction_x == 'right' AND $x1-abs($sx) < $from[0])
599 // $x1 = $to[0]-$width;
602 // if (($direction_y == 'down' AND !$switch) OR ($direction_y == 'up' AND !$switch))
603 // { // switch to the bottom border
604 // $gap = $y1-$from[1]; // vertical gap
605 // $y2 = $to[1]-$gap;
606 // $y1 = $y2-$height;
608 // elseif (($direction_y == 'down' AND $switch) OR ($direction_y == 'up' AND !$switch))
609 // { // switch to the upper border
610 // $gap = $to[1]-$y2; // vertical gap
611 // $y1 = $from[0]+$gap;
612 // $y2 = $y1+$height;
615 // if (round($gap) == round($height/2)) // we are at the vertical center and nothing has been found
618 // if ($switch) // switch to the other side (switching between top and down and going to the vertical center)
623 // $switch = !$switch;
631 // // if we don't have a way to go -- this is useless now when searching from borders to the vertical center
632 // // if (($direction_y == 'up' AND $y1 <= $from[1]) OR ($direction_y == 'down' AND $y2 >= $to[1]))
640 // // if ($intersect) // nothing found
643 // return array($x1, $y1, $x2, $y2);