BlueMap/BlueMapCore/src/main/java/de/bluecolored/bluemap/core/util/math/Color.java

216 lines
6.6 KiB
Java

/*
* This file is part of BlueMap, licensed under the MIT License (MIT).
*
* Copyright (c) Blue (Lukas Rieger) <https://bluecolored.de>
* Copyright (c) contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package de.bluecolored.bluemap.core.util.math;
import de.bluecolored.bluemap.api.debug.DebugDump;
@SuppressWarnings("UnusedReturnValue")
@DebugDump
public class Color {
public float r, g, b, a;
public boolean premultiplied;
public Color set(float r, float g, float b, float a, boolean premultiplied) {
this.r = r;
this.g = g;
this.b = b;
this.a = a;
this.premultiplied = premultiplied;
return this;
}
public Color set(Color color) {
this.r = color.r;
this.g = color.g;
this.b = color.b;
this.a = color.a;
this.premultiplied = color.premultiplied;
return this;
}
public Color set(int color) {
return set(color, false);
}
public Color set(int color, boolean premultiplied) {
this.r = ((color >> 16) & 0xFF) / 255f;
this.g = ((color >> 8) & 0xFF) / 255f;
this.b = (color & 0xFF) / 255f;
this.a = ((color >> 24) & 0xFF) / 255f;
this.premultiplied = premultiplied;
return this;
}
public int getInt() {
int r = (int) (this.r * 255) & 0xFF;
int g = (int) (this.g * 255) & 0xFF;
int b = (int) (this.b * 255) & 0xFF;
int a = (int) (this.a * 255) & 0xFF;
return (a << 24) | (r << 16) | (g << 8) | b;
}
public Color add(Color color) {
if (color.a < 1f && !color.premultiplied){
throw new IllegalArgumentException("Can only add premultiplied colors with alpha!");
}
premultiplied();
this.r += color.r;
this.g += color.g;
this.b += color.b;
this.a += color.a;
return this;
}
public Color div(int divisor) {
premultiplied();
float p = 1f / divisor;
this.r *= p;
this.g *= p;
this.b *= p;
this.a *= p;
return this;
}
public Color multiply(Color color) {
if (color.premultiplied) premultiplied();
else straight();
this.r *= color.r;
this.g *= color.g;
this.b *= color.b;
this.a *= color.a;
return this;
}
public Color overlay(Color color) {
if (color.a < 1f && !color.premultiplied) throw new IllegalArgumentException("Can only overlay premultiplied colors with alpha!");
premultiplied();
float p = 1 - color.a;
this.a = p * this.a + color.a;
this.r = p * this.r + color.r;
this.g = p * this.g + color.g;
this.b = p * this.b + color.b;
return this;
}
public Color underlay(Color color) {
if (color.a < 1f && !color.premultiplied) throw new IllegalArgumentException("Can only underlay premultiplied colors with alpha!");
premultiplied();
float p = 1 - a;
this.a = p * color.a + this.a;
this.r = p * color.r + this.r;
this.g = p * color.g + this.g;
this.b = p * color.b + this.b;
return this;
}
public Color flatten() {
if (this.a == 1f) return this;
if (premultiplied && this.a > 0f) {
float m = 1f / this.a;
this.r *= m;
this.g *= m;
this.b *= m;
}
this.a = 1f;
return this;
}
public Color premultiplied() {
if (!premultiplied) {
this.r *= this.a;
this.g *= this.a;
this.b *= this.a;
this.premultiplied = true;
}
return this;
}
public Color straight() {
if (premultiplied) {
if (this.a > 0f) {
float m = 1f / this.a;
this.r *= m;
this.g *= m;
this.b *= m;
}
this.premultiplied = false;
}
return this;
}
/**
* Parses the color from a string and sets it to this Color instance.
* The value can be an integer in String-Format or a string in hexadecimal format prefixed with # (css-style: e.g. #f16 becomes #ff1166).
* @param value The String to parse
* @return The parsed Integer
* @throws NumberFormatException If the value is not formatted correctly or if there is no value present.
*/
public Color parse(String value) {
String val = value;
if (val.charAt(0) == '#') {
val = val.substring(1);
if (val.length() == 3) val = val + "f";
if (val.length() == 4) val = "" + val.charAt(0) + val.charAt(0) + val.charAt(1) + val.charAt(1) + val.charAt(2) + val.charAt(2) + val.charAt(3) + val.charAt(3);
if (val.length() == 6) val = val + "ff";
if (val.length() != 8) throw new NumberFormatException("Invalid color format: '" + value + "'!");
val = val.substring(6, 8) + val.substring(0, 6); // move alpha to front
return set(Integer.parseUnsignedInt(val, 16));
}
int color = Integer.parseInt(val);
if ((color & 0xFF000000) == 0) color |= 0xFF000000; // assume full alpha if not present
return set(color);
}
@Override
public String toString() {
return "Color{" +
"r=" + r + " (" + (int) (r * 255) + ")" +
", g=" + g + " (" + (int) (g * 255) + ")" +
", b=" + b + " (" + (int) (b * 255) + ")" +
", a=" + a + " (" + (int) (a * 255) + ")" +
", premultiplied=" + premultiplied +
'}';
}
}