1/*
2 * Copyright (C) 2006 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.graphics;
18
19import android.annotation.ColorInt;
20import android.annotation.Size;
21import android.util.MathUtils;
22import com.android.internal.util.XmlUtils;
23
24import java.util.HashMap;
25import java.util.Locale;
26
27/**
28 * The Color class defines methods for creating and converting color ints.
29 * Colors are represented as packed ints, made up of 4 bytes: alpha, red,
30 * green, blue. The values are unpremultiplied, meaning any transparency is
31 * stored solely in the alpha component, and not in the color components. The
32 * components are stored as follows (alpha << 24) | (red << 16) |
33 * (green << 8) | blue. Each component ranges between 0..255 with 0
34 * meaning no contribution for that component, and 255 meaning 100%
35 * contribution. Thus opaque-black would be 0xFF000000 (100% opaque but
36 * no contributions from red, green, or blue), and opaque-white would be
37 * 0xFFFFFFFF
38 */
39public class Color {
40    @ColorInt public static final int BLACK       = 0xFF000000;
41    @ColorInt public static final int DKGRAY      = 0xFF444444;
42    @ColorInt public static final int GRAY        = 0xFF888888;
43    @ColorInt public static final int LTGRAY      = 0xFFCCCCCC;
44    @ColorInt public static final int WHITE       = 0xFFFFFFFF;
45    @ColorInt public static final int RED         = 0xFFFF0000;
46    @ColorInt public static final int GREEN       = 0xFF00FF00;
47    @ColorInt public static final int BLUE        = 0xFF0000FF;
48    @ColorInt public static final int YELLOW      = 0xFFFFFF00;
49    @ColorInt public static final int CYAN        = 0xFF00FFFF;
50    @ColorInt public static final int MAGENTA     = 0xFFFF00FF;
51    @ColorInt public static final int TRANSPARENT = 0;
52
53    /**
54     * Return the alpha component of a color int. This is the same as saying
55     * color >>> 24
56     */
57    public static int alpha(int color) {
58        return color >>> 24;
59    }
60
61    /**
62     * Return the red component of a color int. This is the same as saying
63     * (color >> 16) & 0xFF
64     */
65    public static int red(int color) {
66        return (color >> 16) & 0xFF;
67    }
68
69    /**
70     * Return the green component of a color int. This is the same as saying
71     * (color >> 8) & 0xFF
72     */
73    public static int green(int color) {
74        return (color >> 8) & 0xFF;
75    }
76
77    /**
78     * Return the blue component of a color int. This is the same as saying
79     * color & 0xFF
80     */
81    public static int blue(int color) {
82        return color & 0xFF;
83    }
84
85    /**
86     * Return a color-int from red, green, blue components.
87     * The alpha component is implicity 255 (fully opaque).
88     * These component values should be [0..255], but there is no
89     * range check performed, so if they are out of range, the
90     * returned color is undefined.
91     * @param red  Red component [0..255] of the color
92     * @param green Green component [0..255] of the color
93     * @param blue  Blue component [0..255] of the color
94     */
95    @ColorInt
96    public static int rgb(int red, int green, int blue) {
97        return (0xFF << 24) | (red << 16) | (green << 8) | blue;
98    }
99
100    /**
101     * Return a color-int from alpha, red, green, blue components.
102     * These component values should be [0..255], but there is no
103     * range check performed, so if they are out of range, the
104     * returned color is undefined.
105     * @param alpha Alpha component [0..255] of the color
106     * @param red   Red component [0..255] of the color
107     * @param green Green component [0..255] of the color
108     * @param blue  Blue component [0..255] of the color
109     */
110    @ColorInt
111    public static int argb(int alpha, int red, int green, int blue) {
112        return (alpha << 24) | (red << 16) | (green << 8) | blue;
113    }
114
115    /**
116     * Returns the relative luminance of a color.
117     * <p>
118     * Assumes sRGB encoding. Based on the formula for relative luminance
119     * defined in WCAG 2.0, W3C Recommendation 11 December 2008.
120     *
121     * @return a value between 0 (darkest black) and 1 (lightest white)
122     */
123    public static float luminance(@ColorInt int color) {
124        double red = Color.red(color) / 255.0;
125        red = red < 0.03928 ? red / 12.92 : Math.pow((red + 0.055) / 1.055, 2.4);
126        double green = Color.green(color) / 255.0;
127        green = green < 0.03928 ? green / 12.92 : Math.pow((green + 0.055) / 1.055, 2.4);
128        double blue = Color.blue(color) / 255.0;
129        blue = blue < 0.03928 ? blue / 12.92 : Math.pow((blue + 0.055) / 1.055, 2.4);
130        return (float) ((0.2126 * red) + (0.7152 * green) + (0.0722 * blue));
131    }
132
133    /**
134     * Parse the color string, and return the corresponding color-int.
135     * If the string cannot be parsed, throws an IllegalArgumentException
136     * exception. Supported formats are:
137     * #RRGGBB
138     * #AARRGGBB
139     * or one of the following names:
140     * 'red', 'blue', 'green', 'black', 'white', 'gray', 'cyan', 'magenta',
141     * 'yellow', 'lightgray', 'darkgray', 'grey', 'lightgrey', 'darkgrey',
142     * 'aqua', 'fuchsia', 'lime', 'maroon', 'navy', 'olive', 'purple',
143     * 'silver', 'teal'.
144     */
145    @ColorInt
146    public static int parseColor(@Size(min=1) String colorString) {
147        if (colorString.charAt(0) == '#') {
148            // Use a long to avoid rollovers on #ffXXXXXX
149            long color = Long.parseLong(colorString.substring(1), 16);
150            if (colorString.length() == 7) {
151                // Set the alpha value
152                color |= 0x00000000ff000000;
153            } else if (colorString.length() != 9) {
154                throw new IllegalArgumentException("Unknown color");
155            }
156            return (int)color;
157        } else {
158            Integer color = sColorNameMap.get(colorString.toLowerCase(Locale.ROOT));
159            if (color != null) {
160                return color;
161            }
162        }
163        throw new IllegalArgumentException("Unknown color");
164    }
165
166    /**
167     * Convert RGB components to HSV.
168     *     hsv[0] is Hue [0 .. 360)
169     *     hsv[1] is Saturation [0...1]
170     *     hsv[2] is Value [0...1]
171     * @param red  red component value [0..255]
172     * @param green  green component value [0..255]
173     * @param blue  blue component value [0..255]
174     * @param hsv  3 element array which holds the resulting HSV components.
175     */
176    public static void RGBToHSV(int red, int green, int blue, @Size(3) float hsv[]) {
177        if (hsv.length < 3) {
178            throw new RuntimeException("3 components required for hsv");
179        }
180        nativeRGBToHSV(red, green, blue, hsv);
181    }
182
183    /**
184     * Convert the argb color to its HSV components.
185     *     hsv[0] is Hue [0 .. 360)
186     *     hsv[1] is Saturation [0...1]
187     *     hsv[2] is Value [0...1]
188     * @param color the argb color to convert. The alpha component is ignored.
189     * @param hsv  3 element array which holds the resulting HSV components.
190     */
191    public static void colorToHSV(@ColorInt int color, @Size(3) float hsv[]) {
192        RGBToHSV((color >> 16) & 0xFF, (color >> 8) & 0xFF, color & 0xFF, hsv);
193    }
194
195    /**
196     * Convert HSV components to an ARGB color. Alpha set to 0xFF.
197     *     hsv[0] is Hue [0 .. 360)
198     *     hsv[1] is Saturation [0...1]
199     *     hsv[2] is Value [0...1]
200     * If hsv values are out of range, they are pinned.
201     * @param hsv  3 element array which holds the input HSV components.
202     * @return the resulting argb color
203    */
204    public static int HSVToColor(@Size(3) float hsv[]) {
205        return HSVToColor(0xFF, hsv);
206    }
207
208    /**
209     * Convert HSV components to an ARGB color. The alpha component is passed
210     * through unchanged.
211     *     hsv[0] is Hue [0 .. 360)
212     *     hsv[1] is Saturation [0...1]
213     *     hsv[2] is Value [0...1]
214     * If hsv values are out of range, they are pinned.
215     * @param alpha the alpha component of the returned argb color.
216     * @param hsv  3 element array which holds the input HSV components.
217     * @return the resulting argb color
218    */
219    public static int HSVToColor(int alpha, @Size(3) float hsv[]) {
220        if (hsv.length < 3) {
221            throw new RuntimeException("3 components required for hsv");
222        }
223        return nativeHSVToColor(alpha, hsv);
224    }
225
226    private static native void nativeRGBToHSV(int red, int greed, int blue, float hsv[]);
227    private static native int nativeHSVToColor(int alpha, float hsv[]);
228
229    /**
230     * Converts an HTML color (named or numeric) to an integer RGB value.
231     *
232     * @param color Non-null color string.
233     *
234     * @return A color value, or {@code -1} if the color string could not be interpreted.
235     *
236     * @hide
237     */
238    @ColorInt
239    public static int getHtmlColor(String color) {
240        Integer i = sColorNameMap.get(color.toLowerCase(Locale.ROOT));
241        if (i != null) {
242            return i;
243        } else {
244            try {
245                return XmlUtils.convertValueToInt(color, -1);
246            } catch (NumberFormatException nfe) {
247                return -1;
248            }
249        }
250    }
251
252    private static final HashMap<String, Integer> sColorNameMap;
253
254    static {
255        sColorNameMap = new HashMap<String, Integer>();
256        sColorNameMap.put("black", BLACK);
257        sColorNameMap.put("darkgray", DKGRAY);
258        sColorNameMap.put("gray", GRAY);
259        sColorNameMap.put("lightgray", LTGRAY);
260        sColorNameMap.put("white", WHITE);
261        sColorNameMap.put("red", RED);
262        sColorNameMap.put("green", GREEN);
263        sColorNameMap.put("blue", BLUE);
264        sColorNameMap.put("yellow", YELLOW);
265        sColorNameMap.put("cyan", CYAN);
266        sColorNameMap.put("magenta", MAGENTA);
267        sColorNameMap.put("aqua", 0xFF00FFFF);
268        sColorNameMap.put("fuchsia", 0xFFFF00FF);
269        sColorNameMap.put("darkgrey", DKGRAY);
270        sColorNameMap.put("grey", GRAY);
271        sColorNameMap.put("lightgrey", LTGRAY);
272        sColorNameMap.put("lime", 0xFF00FF00);
273        sColorNameMap.put("maroon", 0xFF800000);
274        sColorNameMap.put("navy", 0xFF000080);
275        sColorNameMap.put("olive", 0xFF808000);
276        sColorNameMap.put("purple", 0xFF800080);
277        sColorNameMap.put("silver", 0xFFC0C0C0);
278        sColorNameMap.put("teal", 0xFF008080);
279
280    }
281}
282