'use strict';
const Rac = require('../Rac');
const utils = require('../util/utils');
/**
* Color with RBGA values, each one on the *[0,1]* range.
*
* @alias Rac.Color
*/
class Color {
/**
* Creates a new `Color` instance.
*
* @param {Rac} rac - Instance to use for drawing and creating other objects
* @param {number} r - The red channel value, in the *[0,1]* range
* @param {number} g - The green channel value, in the *[0,1]* range
* @param {number} b - The blue channel value, in the *[0,1]* range
* @param {number} [a=1] - The alpha channel value, in the *[0,1]* range
*/
constructor(rac, r, g, b, a = 1) {
utils.assertExists(rac, r, g, b, a);
utils.assertNumber(r, g, b, a);
/**
* Instance of `Rac` used for drawing and passed along to any created
* object.
*
* @type {Rac}
*/
this.rac = rac;
/**
* The red channel of the color, in the *[0,1]* range.
* @type {number}
*/
this.r = r;
/**
* The green channel of the color, in the *[0,1]* range.
* @type {number}
*/
this.g = g;
/**
* The blue channel of the color, in the *[0,1]* range.
* @type {number}
*/
this.b = b;
/**
* The alpha channel of the color, in the *[0,1]* range.
* @type {number}
*/
this.a = a;
}
/**
* Returns a string representation intended for human consumption.
*
* @returns {string}
*/
toString() {
return `Color(${this.r},${this.g},${this.b},${this.a})`;
}
/**
* Creates a new `Color` instance with each channel received in the
* *[0,255]* range
*
* @param {Rac} rac - Instance to use for drawing and creating other objects
* @param {number} r - The red channel value, in the *[0,255]* range
* @param {number} g - The green channel value, in the *[0,255]* range
* @param {number} b - The blue channel value, in the *[0,255]* range
* @param {number} [a=255] - The alpha channel value, in the *[0,255]* range
*
* @returns {Rac.Color}
*/
static fromRgba(rac, r, g, b, a = 255) {
return new Color(rac, r/255, g/255, b/255, a/255);
}
/**
* Creates a new `Color` instance from a hexadecimal triplet string.
*
* The `hexString` is expected to have 6 digits and can optionally start
* with `#`. `AABBCC` and `#DDEEFF` are both valid inputs, the three digit
* shorthand is not yet supported.
*
* An error is thrown if `hexString` is misformatted or cannot be parsed.
*
* @param {Rac} rac - Instance to use for drawing and creating other objects
* @param {string} hexString - The RGB hex triplet to interpret
*
* @returns {Rac.Color}
*/
static fromHex(rac, hexString) {
if (hexString.charAt(0) == '#') {
hexString = hexString.substring(1);
}
if (hexString.length != 6) {
throw Rac.Exception.failedAssert(
`Unexpected length for hex triplet string: ${hexString}`);
}
let rStr = hexString.substring(0, 2);
let gStr = hexString.substring(2, 4);
let bStr = hexString.substring(4, 6);
let newR = parseInt(rStr, 16);
let newG = parseInt(gStr, 16);
let newB = parseInt(bStr, 16);
if (isNaN(newR) || isNaN(newG) || isNaN(newB)) {
throw Rac.Exception.failedAssert(
`Could not parse hex triplet string: ${hexString}`);
}
return new Color(rac, newR/255, newG/255, newB/255);
}
/**
* Returns a new `Fill` that uses `this` as `color`.
*
* @returns {Rac.Fill}
*/
fill() {
return new Rac.Fill(this.rac, this);
}
/**
* Returns a new `Stroke` that uses `this` as `color`.
*
* @param {?number} weight - The weight of the new `Stroke`
* @returns {Rac.Stroke}
*/
stroke(weight = null) {
return new Rac.Stroke(this.rac, weight, this);
}
/**
* Returns a new `Color` with `a` set to `newAlpha`.
*
* @param {number} newAlpha - The alpha channel for the new `Color`, in the
* *[0,1]* range
* @returns {Rac.Color}
*/
withAlpha(newAlpha) {
return new Color(this.rac, this.r, this.g, this.b, newAlpha);
}
/**
* Returns a new `Color` with `a` set to `this.a * ratio`.
*
* @param {number} ratio - The factor to multiply `a` by
* @returns {Rac.Color}
*/
withAlphaRatio(ratio) {
return new Color(this.rac, this.r, this.g, this.b, this.a * ratio);
}
/**
* Returns a new `Color` in the linear transition between `this` and
* `target` at a `ratio` in the range *[0,1]*.
*
* When `ratio` is `0` or less the new `Color` is equivalent to `this`,
* when `ratio` is `1` or larger the new `Color` is equivalent to
* `target`.
*
* @param {number} ratio - The transition ratio for the new `Color`
* @param {Rac.Color} target - The transition target `Color`
* @returns {Rac.Color}
*/
linearTransition(ratio, target) {
ratio = Math.max(ratio, 0);
ratio = Math.min(ratio, 1);
let newR = this.r + (target.r - this.r) * ratio;
let newG = this.g + (target.g - this.g) * ratio;
let newB = this.b + (target.b - this.b) * ratio;
let newA = this.a + (target.a - this.a) * ratio;
return new Color(this.rac, newR, newG, newB, newA);
}
} // class Color
module.exports = Color;