icingaweb2/library/Icinga/Chart/Graph/LineGraph.php

202 lines
5.6 KiB
PHP

<?php
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
namespace Icinga\Chart\Graph;
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;
/**
* LineGraph implementation for drawing a set of datapoints as
* a connected path
*/
class LineGraph extends Styleable implements Drawable
{
/**
* The dataset to use
*
* @var array
*/
private $dataset;
/**
* True to show dots for each datapoint
*
* @var bool
*/
private $showDataPoints = false;
/**
* When true, the path will be discrete, i.e. showing hard steps instead of a direct line
*
* @var bool
*/
private $isDiscrete = false;
/**
* The tooltips
*
* @var
*/
private $tooltips;
/**
* The default stroke width
* @var int
*/
public $strokeWidth = 5;
/**
* The size of the displayed dots
*
* @var int
*/
public $dotWith = 0;
/**
* Create a new LineGraph displaying the given dataset
*
* @param array $dataset An array of [x, y] arrays to display
*/
public function __construct(
array $dataset,
array &$graphs,
$order,
array $tooltips = null
) {
usort($dataset, array($this, 'sortByX'));
$this->dataset = $dataset;
$this->graphs = $graphs;
$this->tooltips = $tooltips;
foreach ($this->tooltips as $value) {
$ts[] = $value;
}
$this->tooltips = $ts;
$this->order = $order;
}
/**
* Set datapoints to be emphased via dots
*
* @param bool $bool True to enable datapoints, otherwise false
*/
public function setShowDataPoints($bool)
{
$this->showDataPoints = $bool;
}
/**
* Sort the daset by the xaxis
*
* @param array $v1
* @param array $v2
* @return int
*/
private function sortByX(array $v1, array $v2)
{
if ($v1[0] === $v2[0]) {
return 0;
}
return ($v1[0] < $v2[0]) ? -1 : 1;
}
/**
* Configure this style
*
* @param array $cfg The configuration as given in the drawLine call
*/
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
*/
public function toSvg(RenderContext $ctx)
{
$path = new Path($this->dataset);
if ($this->isDiscrete) {
$path->setDiscrete(true);
}
$path->setStrokeColor($this->strokeColor);
$path->setStrokeWidth($this->strokeWidth);
$path->setAttribute('data-icinga-graph-type', 'line');
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);
}
$path->setAdditionalStyle('clip-path: url(#clip);');
$path->setId($this->id);
$group = $path->toSvg($ctx);
foreach ($this->dataset as $x => $point) {
if ($this->showDataPoints === true) {
$dot = new Circle($point[0], $point[1], $this->dotWith);
$dot->setFill($this->strokeColor);
$group->appendChild($dot->toSvg($ctx));
}
// Draw invisible circle for tooltip hovering
$invisible = new Circle($point[0], $point[1], 20);
$invisible->setFill($this->strokeColor);
$invisible->setAdditionalStyle('opacity: 0.0;');
$invisible->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;
$invisible->setAttribute(
'title',
$this->tooltips[$x]->renderNoHtml($this->order, $data, $format)
);
$invisible->setAttribute(
'data-title-rich',
$this->tooltips[$x]->render($this->order, $data, $format)
);
}
$group->appendChild($invisible->toSvg($ctx));
}
return $group;
}
}