2013-09-09 13:55:29 +02:00
|
|
|
<?php
|
2015-02-04 10:46:36 +01:00
|
|
|
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
2013-09-09 13:55:29 +02:00
|
|
|
|
|
|
|
namespace Icinga\Chart\Graph;
|
|
|
|
|
2014-07-09 18:04:03 +02:00
|
|
|
use DOMElement;
|
|
|
|
use Icinga\Chart\Primitive\Animation;
|
|
|
|
use Icinga\Chart\Primitive\Drawable;
|
|
|
|
use Icinga\Chart\Primitive\Rect;
|
|
|
|
use Icinga\Chart\Primitive\Styleable;
|
|
|
|
use Icinga\Chart\Render\RenderContext;
|
2013-09-09 13:55:29 +02:00
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* Bar graph implementation
|
|
|
|
*/
|
2013-09-09 13:55:29 +02:00
|
|
|
class BarGraph extends Styleable implements Drawable
|
|
|
|
{
|
2014-09-02 12:24:29 +02:00
|
|
|
/**
|
|
|
|
* The dataset order
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
private $order = 0;
|
|
|
|
|
2014-02-18 18:35:34 +01:00
|
|
|
/**
|
|
|
|
* The width of the bars.
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
*/
|
2015-01-19 15:12:00 +01:00
|
|
|
private $barWidth = 3;
|
2014-02-18 18:35:34 +01:00
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* The dataset to use for this bar graph
|
2013-09-25 16:32:28 +02:00
|
|
|
*
|
2013-09-21 17:35:18 +02:00
|
|
|
* @var array
|
|
|
|
*/
|
2013-09-09 13:55:29 +02:00
|
|
|
private $dataSet;
|
2013-09-21 17:35:18 +02:00
|
|
|
|
2014-09-02 12:24:29 +02:00
|
|
|
/**
|
|
|
|
* The tooltips
|
|
|
|
*
|
|
|
|
* @var
|
|
|
|
*/
|
|
|
|
private $tooltips;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* All graphs
|
|
|
|
*
|
|
|
|
* @var
|
|
|
|
*/
|
|
|
|
private $graphs;
|
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* Create a new BarGraph with the given dataset
|
|
|
|
*
|
2014-09-02 12:24:29 +02:00
|
|
|
* @param array $dataSet An array of data points
|
|
|
|
* @param int $order The graph number displayed by this BarGraph
|
|
|
|
* @param array $tooltips The tooltips to display for each value
|
2013-09-21 17:35:18 +02:00
|
|
|
*/
|
2014-09-02 12:24:29 +02:00
|
|
|
public function __construct(
|
|
|
|
array $dataSet,
|
|
|
|
array &$graphs,
|
|
|
|
$order,
|
|
|
|
array $tooltips = null
|
|
|
|
) {
|
|
|
|
$this->order = $order;
|
2013-09-09 13:55:29 +02:00
|
|
|
$this->dataSet = $dataSet;
|
2015-02-18 12:55:33 +01:00
|
|
|
|
2014-09-02 12:24:29 +02:00
|
|
|
$this->tooltips = $tooltips;
|
2015-02-18 12:55:33 +01:00
|
|
|
foreach ($this->tooltips as $value) {
|
|
|
|
$ts[] = $value;
|
|
|
|
}
|
|
|
|
$this->tooltips = $ts;
|
|
|
|
|
2014-09-02 12:24:29 +02:00
|
|
|
$this->graphs = $graphs;
|
2013-09-09 13:55:29 +02:00
|
|
|
}
|
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* Apply configuration styles from the $cfg
|
|
|
|
*
|
|
|
|
* @param array $cfg The configuration as given in the drawBars call
|
|
|
|
*/
|
|
|
|
public function setStyleFromConfig(array $cfg)
|
|
|
|
{
|
|
|
|
foreach ($cfg as $elem => $value) {
|
|
|
|
if ($elem === 'color') {
|
|
|
|
$this->setFill($value);
|
|
|
|
} elseif ($elem === 'width') {
|
|
|
|
$this->setStrokeWidth($value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-02 12:24:29 +02:00
|
|
|
/**
|
|
|
|
* Draw a single rectangle
|
|
|
|
*
|
|
|
|
* @param array $point The
|
|
|
|
* @param null $index
|
|
|
|
* @param string $fill The fill color to use
|
|
|
|
* @param $strokeWidth
|
|
|
|
*
|
|
|
|
* @return Rect
|
|
|
|
*/
|
|
|
|
private function drawSingleBar($point, $index = null, $fill, $strokeWidth)
|
|
|
|
{
|
|
|
|
$rect = new Rect($point[0] - ($this->barWidth / 2), $point[1], $this->barWidth, 100 - $point[1]);
|
|
|
|
$rect->setFill($fill);
|
|
|
|
$rect->setStrokeWidth($strokeWidth);
|
|
|
|
$rect->setStrokeColor('black');
|
|
|
|
if (isset($index)) {
|
|
|
|
$rect->setAttribute('data-icinga-graph-index', $index);
|
|
|
|
}
|
|
|
|
$rect->setAttribute('data-icinga-graph-type', 'bar');
|
|
|
|
$rect->setAdditionalStyle('clip-path: url(#clip);');
|
|
|
|
return $rect;
|
|
|
|
}
|
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* Render this BarChart
|
|
|
|
*
|
|
|
|
* @param RenderContext $ctx The rendering context to use for drawing
|
|
|
|
*
|
|
|
|
* @return DOMElement $dom Element
|
|
|
|
*/
|
2013-09-09 13:55:29 +02:00
|
|
|
public function toSvg(RenderContext $ctx)
|
|
|
|
{
|
|
|
|
$doc = $ctx->getDocument();
|
|
|
|
$group = $doc->createElement('g');
|
2013-09-23 12:09:40 +02:00
|
|
|
$idx = 0;
|
2015-01-19 15:12:00 +01:00
|
|
|
|
|
|
|
if (count($this->dataSet) > 15) {
|
|
|
|
$this->barWidth = 2;
|
|
|
|
}
|
|
|
|
if (count($this->dataSet) > 25) {
|
|
|
|
$this->barWidth = 1;
|
|
|
|
}
|
|
|
|
|
2014-09-02 12:24:29 +02:00
|
|
|
foreach ($this->dataSet as $x => $point) {
|
|
|
|
// add white background bar, to prevent other bars from altering transparency effects
|
|
|
|
$bar = $this->drawSingleBar($point, $idx++, 'white', $this->strokeWidth, $idx)->toSvg($ctx);
|
|
|
|
$group->appendChild($bar);
|
|
|
|
|
|
|
|
// draw actual bar
|
|
|
|
$bar = $this->drawSingleBar($point, null, $this->fill, $this->strokeWidth, $idx)->toSvg($ctx);
|
|
|
|
$bar->setAttribute('class', 'chart-data');
|
|
|
|
if (isset($this->tooltips[$x])) {
|
|
|
|
$data = array(
|
|
|
|
'label' => isset($this->graphs[$this->order]['label']) ?
|
|
|
|
strtolower($this->graphs[$this->order]['label']) : '',
|
|
|
|
'color' => isset($this->graphs[$this->order]['color']) ?
|
|
|
|
strtolower($this->graphs[$this->order]['color']) : '#fff'
|
|
|
|
);
|
|
|
|
$format = isset($this->graphs[$this->order]['tooltip'])
|
|
|
|
? $this->graphs[$this->order]['tooltip'] : null;
|
|
|
|
$bar->setAttribute(
|
|
|
|
'title',
|
|
|
|
$this->tooltips[$x]->renderNoHtml($this->order, $data, $format)
|
|
|
|
);
|
|
|
|
$bar->setAttribute(
|
2014-09-04 13:32:30 +02:00
|
|
|
'data-title-rich',
|
2014-09-02 12:24:29 +02:00
|
|
|
$this->tooltips[$x]->render($this->order, $data, $format)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
$group->appendChild($bar);
|
2013-09-09 13:55:29 +02:00
|
|
|
}
|
|
|
|
return $group;
|
|
|
|
}
|
2013-09-21 17:35:18 +02:00
|
|
|
}
|