2021-05-21 08:49:41 +02:00

226 lines
5.5 KiB
PHP

<?php
namespace Xentral\Widgets\Chart;
use InvalidArgumentException;
use JsonSerializable;
class Color implements JsonSerializable
{
private $red;
private $green;
private $blue;
private $alpha;
/**
* @param int $red Farbwert von 0 bis 255
* @param int $green Farbwert von 0 bis 255
* @param int $blue Farbwert von 0 bis 255
* @param float $alpha Transparenzwert von 0 bis 1
*/
public function __construct($red = 0, $green = 0, $blue = 0, $alpha = 0.1)
{
$this->red = $this->ensureColorValue($red);
$this->green = $this->ensureColorValue($green);
$this->blue = $this->ensureColorValue($blue);
$this->setAlpha($alpha);
}
/**
* @param string $hexColor Beispiel: "#112233"
*
* @return self
*/
public static function createFromHex($hexColor)
{
$hexColor = str_replace('#', '', $hexColor);
if (strlen($hexColor) !== 6) {
throw new InvalidArgumentException('Only full length hex values are supported.');
}
$parts = str_split($hexColor, 2);
$red = hexdec($parts[0]);
$green = hexdec($parts[1]);
$blue = hexdec($parts[2]);
return new self($red, $green, $blue, 1);
}
/**
* @param string $cssRgba Beispiel: rgba(255, 128, 0, 0.5)
*
* @return self
*/
public static function createFromCssRgba($cssRgba)
{
$cssRgba = str_replace(' ', '', $cssRgba);
preg_match('/^rgba\((\d+),(\d+),(\d+),([\d\.]+)\);?/i', $cssRgba, $colors);
return new self((int)$colors[1], (int)$colors[2], (int)$colors[3], (float)$colors[4]);
}
/**
* @param string $cssRgb Beispiel: "rgb(255, 128, 0)"
*
* @return self
*/
public static function createFromCssRgb($cssRgb)
{
$cssRgb = str_replace(' ', '', $cssRgb);
preg_match('/^rgb\((\d+),(\d+),(\d+)\);?/i', $cssRgb, $colors);
return new self((int)$colors[1], (int)$colors[2], (int)$colors[3], 1.0);
}
/**
* @param int $value
*
* @return int
*/
private function ensureColorValue($value)
{
$value = (int)$value;
if ($value < 0) {
$value = 0;
}
if ($value > 255) {
$value = 255;
}
return $value;
}
/**
* @param float $alpha Wert zwischen 0 und 1
*
* @return void
*/
public function setAlpha($alpha)
{
$alpha = (float)$alpha;
if ($alpha < 0.0) {
$alpha = 0.0;
}
if ($alpha > 1.0) {
$alpha = 1.0;
}
$this->alpha = $alpha;
}
/**
* Farbwerte per Zufall variieren
*
* @param int $difference
*
* @return void
*/
public function makeVariant($difference)
{
$difference = (int)$difference;
$redVariant = $this->red + mt_rand($difference * -1, $difference);
$this->red = $this->ensureColorValue($redVariant);
$greenVariant = $this->green + mt_rand($difference * -1, $difference);
$this->green = $this->ensureColorValue($greenVariant);
$blueVariant = $this->blue + mt_rand($difference * -1, $difference);
$this->blue = $this->ensureColorValue($blueVariant);
}
/**
* Farbe heller machen; verändert nicht die Transparenz
*
* @param float $percent
*
* @return void
*/
public function makeLighter($percent = 10.0)
{
$percent = (float)$percent;
$difference = (int)($percent * 2.55 / 2);
$this->red = $this->ensureColorValue($this->red + $difference);
$this->blue = $this->ensureColorValue($this->blue + $difference);
$this->green = $this->ensureColorValue($this->green + $difference);
}
/**
* Farbe dunkler machen; verändert nicht die Transparenz
*
* @param float $percent
*
* @return void
*/
public function makeDarker($percent = 10.0)
{
$percent = (float)$percent;
$difference = (int)($percent * 2.55 / 2);
$this->red = $this->ensureColorValue($this->red - $difference);
$this->blue = $this->ensureColorValue($this->blue - $difference);
$this->green = $this->ensureColorValue($this->green - $difference);
}
/**
* Ausgabe in CSS rgba() Notation
*
* @return string
*/
public function toCssRgba()
{
return sprintf(
'rgba(%s, %s, %s, %s)',
$this->red,
$this->green,
$this->blue,
number_format($this->alpha, 3, '.', '')
);
}
/**
* Ausgabe in CSS rgb() Notation; Transparenz geht verloren
*
* @return string
*/
public function toCssRgb()
{
return sprintf(
'rgb(%s, %s, %s)',
$this->red,
$this->green,
$this->blue
);
}
/**
* Ausgabe als Hex-Farbwert; Transparenz geht verloren
*
* @return string
*/
public function toHex()
{
return sprintf(
'#%s%s%s',
str_pad(dechex($this->red), 2, '0', STR_PAD_LEFT),
str_pad(dechex($this->green), 2, '0', STR_PAD_LEFT),
str_pad(dechex($this->blue), 2, '0', STR_PAD_LEFT)
);
}
/**
* @return string
*/
public function jsonSerialize()
{
return $this->__toString();
}
/**
* @return string
*/
public function __toString()
{
return $this->toCssRgba();
}
}