All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.jidesoft.utils.ColorUtils Maven / Gradle / Ivy

/*
 * @(#)ColorUtils.java 9/19/2006
 *
 * Copyright 2002 - 2006 JIDE Software Inc. All rights reserved.
 */

package com.jidesoft.utils;

import javax.swing.plaf.ColorUIResource;
import java.awt.*;

/**
 * Several useful methods for Color.
 */
public class ColorUtils {

    /**
     * Gets a derived color from an existing color. The derived color is either lighter or darker version of the given
     * color with the same hue.
     *
     * @param color the given color.
     * @param ratio the ratio. 0.5f if the same color. Any ratio greater than 0.5f will make the result color lighter.
     *              Smaller than 0.5f will make the color darker.
     *
     * @return the derived color.
     */
    public static Color getDerivedColor(Color color, float ratio) {
        if (color != null) {
            float[] hsl = RGBtoHSL(color);
            if (hsl[2] < 0.4) {
                hsl[2] = 0.4f;
            }
            if (ratio > 0.5) {
                hsl[2] += (1f - hsl[2]) * 2 * (ratio - 0.5);
            }
            else {
                hsl[2] -= hsl[2] * 2 * (0.5 - ratio);
            }
            int colorRGB = HSLtoRGB(hsl);
            return new ColorUIResource(colorRGB);
        }
        else {
            return null;
        }
    }

    /**
     * Converts a color from RBG to HSL color space.
     *
     * @param colorRGB the Color.
     *
     * @return color space in HSL.
     */
    public static float[] RGBtoHSL(Color colorRGB) {
        float r, g, b, h, s, l; //this function works with floats between 0 and 1
        r = colorRGB.getRed() / 256.0f;
        g = colorRGB.getGreen() / 256.0f;
        b = colorRGB.getBlue() / 256.0f;

        // Then, minColor and maxColor are defined. Min color is the value of the color component with
        // the smallest value, while maxColor is the value of the color component with the largest value.
        // These two variables are needed because the Lightness is defined as (minColor + maxColor) / 2.

        float maxColor = Math.max(r, Math.max(g, b));
        float minColor = Math.min(r, Math.min(g, b));

        // If minColor equals maxColor, we know that R=G=B and thus the color is a shade of gray.
        // This is a trivial case, hue can be set to anything, saturation has to be set to 0 because
        // only then it's a shade of gray, and lightness is set to R=G=B, the shade of the gray.

        //R == G == B, so it's a shade of gray
        if (r == g && g == b) {
            h = 0.0f; //it doesn't matter what value it has
            s = 0.0f;
            l = r; //doesn't matter if you pick r, g, or b
        }

        // If minColor is not equal to maxColor, we have a real color instead of a shade of gray, so more calculations are needed:

        // Lightness (l) is now set to it's definition of (minColor + maxColor)/2.
        // Saturation (s) is then calculated with a different formula depending if light is in the first half of the second half. This is because the HSL model can be represented as a double cone, the first cone has a black tip and corresponds to the first half of lightness values, the second cone has a white tip and contains the second half of lightness values.
        // Hue (h) is calculated with a different formula depending on which of the 3 color components is the dominating one, and then normalized to a number between 0 and 1.

        else {
            l = (minColor + maxColor) / 2;

            if (l < 0.5) s = (maxColor - minColor) / (maxColor + minColor);
            else s = (maxColor - minColor) / (2.0f - maxColor - minColor);

            if (r == maxColor) h = (g - b) / (maxColor - minColor);
            else if (g == maxColor) h = 2.0f + (b - r) / (maxColor - minColor);
            else h = 4.0f + (r - g) / (maxColor - minColor);

            h /= 6; //to bring it to a number between 0 and 1
            if (h < 0) h++;
        }

        // Finally, H, S and L are calculated out of h, s and l as integers between 0 and 255 and "returned"
        // as the result. Returned, because H, S and L were passed by reference to the function.

        float[] hsl = new float[3];
        hsl[0] = h;
        hsl[1] = s;
        hsl[2] = l;
        return hsl;
    }

    /**
     * Converts from HSL color space to RGB color.
     *
     * @param hsl the hsl values.
     *
     * @return the RGB color.
     */
    public static int HSLtoRGB(float[] hsl) {
        float r, g, b, h, s, l; //this function works with floats between 0 and 1
        float temp1, temp2, tempr, tempg, tempb;
        h = hsl[0];
        s = hsl[1];
        l = hsl[2];

        // Then follows a trivial case: if the saturation is 0, the color will be a grayscale color,
        // and the calculation is then very simple: r, g and b are all set to the lightness.

        //If saturation is 0, the color is a shade of gray
        if (s == 0) {
            r = g = b = l;
        }
        // If the saturation is higher than 0, more calculations are needed again.
        // red, green and blue are calculated with the formulas defined in the code.
        // If saturation > 0, more complex calculations are needed
        else {
            //Set the temporary values
            if (l < 0.5) temp2 = l * (1 + s);
            else temp2 = (l + s) - (l * s);
            temp1 = 2 * l - temp2;
            tempr = h + 1.0f / 3.0f;
            if (tempr > 1) tempr--;
            tempg = h;
            tempb = h - 1.0f / 3.0f;
            if (tempb < 0) tempb++;

            //Red
            if (tempr < 1.0 / 6.0) r = temp1 + (temp2 - temp1) * 6.0f * tempr;
            else if (tempr < 0.5) r = temp2;
            else if (tempr < 2.0 / 3.0) r = temp1 + (temp2 - temp1) * ((2.0f / 3.0f) - tempr) * 6.0f;
            else r = temp1;

            //Green
            if (tempg < 1.0 / 6.0) g = temp1 + (temp2 - temp1) * 6.0f * tempg;
            else if (tempg < 0.5) g = temp2;
            else if (tempg < 2.0 / 3.0) g = temp1 + (temp2 - temp1) * ((2.0f / 3.0f) - tempg) * 6.0f;
            else g = temp1;

            //Blue
            if (tempb < 1.0 / 6.0) b = temp1 + (temp2 - temp1) * 6.0f * tempb;
            else if (tempb < 0.5) b = temp2;
            else if (tempb < 2.0 / 3.0) b = temp1 + (temp2 - temp1) * ((2.0f / 3.0f) - tempb) * 6.0f;
            else b = temp1;
        }

        // And finally, the results are returned as integers between 0 and 255.

        int result = 0;
        result += ((int) (r * 255) & 0xFF) << 16;
        result += ((int) (g * 255) & 0xFF) << 8;
        result += ((int) (b * 255) & 0xFF);

        return result;
    }

    static final float OFFSET_180 = 180f;
    static final float OFFSET_100 = 100f;

    public static int[] calculateDifferent(float[] from, float[] to) {
        int[] diff = new int[3];
        diff[0] = floatToInteger(from[0], to[0], OFFSET_180, true);
        diff[1] = floatToInteger(from[1], to[1], OFFSET_100, false);
        diff[2] = floatToInteger(from[2], to[2], OFFSET_100, false);
        return diff;
    }

    public static float[] applyDifference(float[] from, int[] diff) {
        float[] to = new float[3];
        to[0] = integerToFloat(from[0], diff[0], OFFSET_180, true);
        to[1] = integerToFloat(from[1], diff[1], OFFSET_100, false);
        to[2] = integerToFloat(from[2], diff[2], OFFSET_100, false);
        return to;
    }

    private static int floatToInteger(float f, float f2, float offset, boolean rotate) {
        if (rotate) {
            int i = (int) ((f2 - f) * 2 * offset);
            if (i > offset) {
                return i - (int) (2 * offset);
            }
            else if (i < -offset) {
                return i + (int) (2 * offset);
            }
            else {
                return i;
            }
        }
        else {
            if (f != 0.0f) {
                return (int) ((f2 - f) * offset / f);
            }
            else {
                return (int) ((f2 - f) * offset);
            }
        }
    }

    private static float integerToFloat(float f, int i, float offset, boolean rotate) {
        if (rotate) {
            float v = f + i / (2 * offset);
            if (v < 0.0f) {
                return v + 1.0f;
            }
            else if (v > 1.0f) {
                return v - 1.0f;
            }
            else {
                return v;
            }
        }
        else {
            if (i > 0) {
                return f + (1.0f - f) * i / offset;
            }
            else {
                return f + f * i / offset;
            }
        }
    }

    /**
     * Simply calls new Color(color, hasalpha) for each color in colors and returns all of them.
     *
     * @param hasAlpha true to consider the alpha when creating the Color.
     * @param colors   the color value.
     *
     * @return the colors with alpha added.
     */
    public static Color[] toColors(boolean hasAlpha, int... colors) {
        Color[] result = new Color[colors.length];
        for (int i = 0; i < colors.length; i++) {
            result[i] = new Color(colors[i], hasAlpha);
        }
        return result;
    }

    /**
     * Converts from a color to gray scale color.
     *
     * @param c a color.
     *
     * @return a color in gray scale.
     */
    public static Color toGrayscale(Color c) {
        int gray = (int) (c.getRed() * 0.3 + c.getGreen() * 0.59 + c.getBlue() * 0.11);
        return new Color(gray, gray, gray);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy