2013-09-09 13:55:29 +02:00
|
|
|
<?php
|
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
|
|
|
|
|
|
|
|
namespace Icinga\Chart\Graph;
|
|
|
|
|
2014-07-09 18:04:03 +02:00
|
|
|
use DOMElement;
|
|
|
|
use Icinga\Chart\Primitive\Drawable;
|
|
|
|
use Icinga\Chart\Primitive\Path;
|
|
|
|
use Icinga\Chart\Primitive\Circle;
|
|
|
|
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
|
|
|
/**
|
|
|
|
* LineGraph implementation for drawing a set of datapoints as
|
|
|
|
* a connected path
|
|
|
|
*/
|
2013-09-09 13:55:29 +02:00
|
|
|
class LineGraph extends Styleable implements Drawable
|
|
|
|
{
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* The dataset to use
|
2013-09-25 16:32:28 +02:00
|
|
|
*
|
2013-09-21 17:35:18 +02:00
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $dataset;
|
2013-09-09 13:55:29 +02:00
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* True to show dots for each datapoint
|
2013-09-25 16:32:28 +02:00
|
|
|
*
|
2013-09-21 17:35:18 +02:00
|
|
|
* @var bool
|
|
|
|
*/
|
2013-09-09 13:55:29 +02:00
|
|
|
private $showDataPoints = false;
|
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* When true, the path will be discrete, i.e. showing hard steps instead of a direct line
|
2013-09-25 16:32:28 +02:00
|
|
|
*
|
2013-09-21 17:35:18 +02:00
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private $isDiscrete = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The default stroke width
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
public $strokeWidth = 5;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new LineGraph displaying the given dataset
|
|
|
|
*
|
2013-09-25 16:32:28 +02:00
|
|
|
* @param array $dataset An array of [x, y] arrays to display
|
2013-09-21 17:35:18 +02:00
|
|
|
*/
|
2013-09-09 13:55:29 +02:00
|
|
|
public function __construct(array $dataset)
|
|
|
|
{
|
|
|
|
usort($dataset, array($this, 'sortByX'));
|
|
|
|
$this->dataset = $dataset;
|
|
|
|
}
|
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* Set datapoints to be emphased via dots
|
|
|
|
*
|
2013-09-25 16:32:28 +02:00
|
|
|
* @param bool $bool True to enable datapoints, otherwise false
|
2013-09-21 17:35:18 +02:00
|
|
|
*/
|
2013-09-09 13:55:29 +02:00
|
|
|
public function setShowDataPoints($bool)
|
|
|
|
{
|
2013-09-21 17:35:18 +02:00
|
|
|
$this->showDataPoints = $bool;
|
2013-09-09 13:55:29 +02:00
|
|
|
}
|
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* Sort the daset by the xaxis
|
|
|
|
*
|
2013-09-25 16:32:28 +02:00
|
|
|
* @param array $v1
|
|
|
|
* @param array $v2
|
|
|
|
* @return int
|
2013-09-21 17:35:18 +02:00
|
|
|
*/
|
|
|
|
private function sortByX(array $v1, array $v2)
|
|
|
|
{
|
|
|
|
if ($v1[0] === $v2[0]) {
|
2013-09-09 13:55:29 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return ($v1[0] < $v2[0]) ? -1 : 1;
|
|
|
|
}
|
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
/**
|
|
|
|
* Configure this style
|
|
|
|
*
|
2013-09-25 16:32:28 +02:00
|
|
|
* @param array $cfg The configuration as given in the drawLine call
|
2013-09-21 17:35:18 +02:00
|
|
|
*/
|
|
|
|
public function setStyleFromConfig(array $cfg)
|
|
|
|
{
|
|
|
|
$fill = false;
|
|
|
|
foreach ($cfg as $elem => $value) {
|
|
|
|
if ($elem === 'color') {
|
|
|
|
$this->setStrokeColor($value);
|
|
|
|
} elseif ($elem === 'width') {
|
|
|
|
$this->setStrokeWidth($value);
|
|
|
|
} elseif ($elem === 'showPoints') {
|
|
|
|
$this->setShowDataPoints($value);
|
|
|
|
} elseif ($elem === 'fill') {
|
|
|
|
$fill = $value;
|
|
|
|
} elseif ($elem === 'discrete') {
|
|
|
|
$this->isDiscrete = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($fill) {
|
|
|
|
$this->setFill($this->strokeColor);
|
|
|
|
$this->setStrokeColor('black');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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)
|
|
|
|
{
|
|
|
|
$path = new Path($this->dataset);
|
2013-09-21 17:35:18 +02:00
|
|
|
if ($this->isDiscrete) {
|
|
|
|
$path->setDiscrete(true);
|
|
|
|
}
|
2013-09-09 13:55:29 +02:00
|
|
|
$path->setStrokeColor($this->strokeColor);
|
|
|
|
$path->setStrokeWidth($this->strokeWidth);
|
2013-09-21 17:35:18 +02:00
|
|
|
|
2013-09-23 12:09:40 +02:00
|
|
|
$path->setAttribute('data-icinga-graph-type', 'line');
|
2013-09-21 17:35:18 +02:00
|
|
|
if ($this->fill !== 'none') {
|
|
|
|
$firstX = $this->dataset[0][0];
|
|
|
|
$lastX = $this->dataset[count($this->dataset)-1][0];
|
|
|
|
$path->prepend(array($firstX, 100))
|
|
|
|
->append(array($lastX, 100));
|
|
|
|
$path->setFill($this->fill);
|
|
|
|
}
|
2013-09-23 12:09:40 +02:00
|
|
|
|
2013-09-21 17:35:18 +02:00
|
|
|
$path->setAdditionalStyle('clip-path: url(#clip);');
|
|
|
|
$path->setId($this->id);
|
2013-09-09 13:55:29 +02:00
|
|
|
$group = $path->toSvg($ctx);
|
|
|
|
if ($this->showDataPoints === true) {
|
|
|
|
foreach ($this->dataset as $point) {
|
2013-09-21 17:35:18 +02:00
|
|
|
$dot = new Circle($point[0], $point[1], $this->strokeWidth*5);
|
|
|
|
$dot->setFill('black');
|
|
|
|
|
2013-09-09 13:55:29 +02:00
|
|
|
$group->appendChild($dot->toSvg($ctx));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $group;
|
|
|
|
}
|
2013-09-21 17:35:18 +02:00
|
|
|
}
|