| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You under the Apache License, Version 2.0 |
| * (the "License"); you may not use this file except in compliance with |
| * the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| /* $Id$ */ |
| |
| package org.apache.fop.util; |
| |
| import java.awt.Color; |
| import java.awt.color.ColorSpace; |
| import java.awt.color.ICC_ColorSpace; |
| import java.awt.color.ICC_Profile; |
| import java.io.IOException; |
| import java.util.Collections; |
| import java.util.Map; |
| |
| import org.apache.commons.logging.Log; |
| import org.apache.commons.logging.LogFactory; |
| |
| import org.apache.xmlgraphics.java2d.color.CIELabColorSpace; |
| import org.apache.xmlgraphics.java2d.color.ColorSpaceOrigin; |
| import org.apache.xmlgraphics.java2d.color.ColorSpaces; |
| import org.apache.xmlgraphics.java2d.color.ColorWithAlternatives; |
| import org.apache.xmlgraphics.java2d.color.DeviceCMYKColorSpace; |
| import org.apache.xmlgraphics.java2d.color.NamedColorSpace; |
| import org.apache.xmlgraphics.java2d.color.RenderingIntent; |
| import org.apache.xmlgraphics.java2d.color.profile.NamedColorProfile; |
| import org.apache.xmlgraphics.java2d.color.profile.NamedColorProfileParser; |
| |
| import org.apache.fop.apps.FOUserAgent; |
| import org.apache.fop.fo.expr.PropertyException; |
| |
| /** |
| * Generic Color helper class. |
| * <p> |
| * This class supports parsing string values into color values and creating |
| * color values for strings. It provides a list of standard color names. |
| */ |
| public final class ColorUtil { |
| |
| //ColorWithFallback is used to preserve the sRGB fallback exclusively for the purpose |
| //of regenerating textual color functions as specified in XSL-FO. |
| |
| /** The name for the uncalibrated CMYK pseudo-profile */ |
| public static final String CMYK_PSEUDO_PROFILE = "#CMYK"; |
| |
| /** The name for the Separation pseudo-profile used for spot colors */ |
| public static final String SEPARATION_PSEUDO_PROFILE = "#Separation"; |
| |
| /** |
| * Keeps all the predefined and parsed colors. |
| * <p> |
| * This map is used to predefine given colors, as well as speeding up |
| * parsing of already parsed colors. |
| * <p> |
| * Important: The use of this color map assumes that all Color instances are immutable! |
| */ |
| private static Map<String, Color> colorMap; |
| |
| /** Logger instance */ |
| protected static final Log log = LogFactory.getLog(ColorUtil.class); |
| |
| static { |
| initializeColorMap(); |
| } |
| |
| /** |
| * Private constructor since this is an utility class. |
| */ |
| private ColorUtil() { |
| } |
| |
| /** |
| * Creates a color from a given string. |
| * <p> |
| * This function supports a wide variety of inputs. |
| * <ul> |
| * <li>#RGB (hex 0..f)</li> |
| * <li>#RGBA (hex 0..f)</li> |
| * <li>#RRGGBB (hex 00..ff)</li> |
| * <li>#RRGGBBAA (hex 00..ff)</li> |
| * <li>rgb(r,g,b) (0..255 or 0%..100%)</li> |
| * <li>java.awt.Color[r=r,g=g,b=b] (0..255)</li> |
| * <li>system-color(colorname)</li> |
| * <li>transparent</li> |
| * <li>colorname</li> |
| * <li>fop-rgb-icc(r,g,b,cs,cs-src,[num]+) (r/g/b: 0..1, num: 0..1)</li> |
| * <li>cmyk(c,m,y,k) (0..1)</li> |
| * </ul> |
| * |
| * @param foUserAgent FOUserAgent object |
| * @param value |
| * the string to parse. |
| * @return a Color representing the string if possible |
| * @throws PropertyException |
| * if the string is not parsable or does not follow any of the |
| * given formats. |
| */ |
| public static Color parseColorString(FOUserAgent foUserAgent, String value) |
| throws PropertyException { |
| if (value == null) { |
| return null; |
| } |
| |
| Color parsedColor = colorMap.get(value.toLowerCase()); |
| |
| if (parsedColor == null) { |
| if (value.startsWith("#")) { |
| parsedColor = parseWithHash(value); |
| } else if (value.startsWith("rgb(")) { |
| parsedColor = parseAsRGB(value); |
| } else if (value.startsWith("url(")) { |
| throw new PropertyException( |
| "Colors starting with url( are not yet supported!"); |
| } else if (value.startsWith("java.awt.Color")) { |
| parsedColor = parseAsJavaAWTColor(value); |
| } else if (value.startsWith("system-color(")) { |
| parsedColor = parseAsSystemColor(value); |
| } else if (value.startsWith("fop-rgb-icc")) { |
| parsedColor = parseAsFopRgbIcc(foUserAgent, value); |
| } else if (value.startsWith("fop-rgb-named-color")) { |
| parsedColor = parseAsFopRgbNamedColor(foUserAgent, value); |
| } else if (value.startsWith("cie-lab-color")) { |
| parsedColor = parseAsCIELabColor(foUserAgent, value); |
| } else if (value.startsWith("cmyk")) { |
| parsedColor = parseAsCMYK(value); |
| } |
| |
| if (parsedColor == null) { |
| throw new PropertyException("Unknown Color: " + value); |
| } |
| |
| colorMap.put(value, parsedColor); |
| } |
| |
| return parsedColor; |
| } |
| |
| /** |
| * Tries to parse a color given with the system-color() function. |
| * |
| * @param value |
| * the complete line |
| * @return a color if possible |
| * @throws PropertyException |
| * if the format is wrong. |
| */ |
| private static Color parseAsSystemColor(String value) |
| throws PropertyException { |
| int poss = value.indexOf("("); |
| int pose = value.indexOf(")"); |
| if (poss != -1 && pose != -1) { |
| value = value.substring(poss + 1, pose); |
| } else { |
| throw new PropertyException("Unknown color format: " + value |
| + ". Must be system-color(x)"); |
| } |
| return colorMap.get(value); |
| } |
| |
| /** |
| * Tries to parse the standard java.awt.Color toString output. |
| * |
| * @param value |
| * the complete line |
| * @return a color if possible |
| * @throws PropertyException |
| * if the format is wrong. |
| * @see java.awt.Color#toString() |
| */ |
| private static Color parseAsJavaAWTColor(String value) |
| throws PropertyException { |
| float red = 0.0f; |
| float green = 0.0f; |
| float blue = 0.0f; |
| int poss = value.indexOf("["); |
| int pose = value.indexOf("]"); |
| try { |
| if (poss != -1 && pose != -1) { |
| value = value.substring(poss + 1, pose); |
| String[] args = value.split(","); |
| if (args.length != 3) { |
| throw new PropertyException( |
| "Invalid number of arguments for a java.awt.Color: " + value); |
| } |
| |
| red = Float.parseFloat(args[0].trim().substring(2)) / 255f; |
| green = Float.parseFloat(args[1].trim().substring(2)) / 255f; |
| blue = Float.parseFloat(args[2].trim().substring(2)) / 255f; |
| if ((red < 0.0 || red > 1.0) |
| || (green < 0.0 || green > 1.0) |
| || (blue < 0.0 || blue > 1.0)) { |
| throw new PropertyException("Color values out of range"); |
| } |
| } else { |
| throw new IllegalArgumentException( |
| "Invalid format for a java.awt.Color: " + value); |
| } |
| } catch (RuntimeException re) { |
| throw new PropertyException(re); |
| } |
| return new Color(red, green, blue); |
| } |
| |
| /** |
| * Parse a color given with the rgb() function. |
| * |
| * @param value |
| * the complete line |
| * @return a color if possible |
| * @throws PropertyException |
| * if the format is wrong. |
| */ |
| private static Color parseAsRGB(String value) throws PropertyException { |
| Color parsedColor; |
| int poss = value.indexOf("("); |
| int pose = value.indexOf(")"); |
| if (poss != -1 && pose != -1) { |
| value = value.substring(poss + 1, pose); |
| try { |
| String[] args = value.split(","); |
| if (args.length != 3) { |
| throw new PropertyException( |
| "Invalid number of arguments: rgb(" + value + ")"); |
| } |
| float red = parseComponent255(args[0], value); |
| float green = parseComponent255(args[1], value); |
| float blue = parseComponent255(args[2], value); |
| //Convert to ints to synchronize the behaviour with toRGBFunctionCall() |
| int r = (int)(red * 255 + 0.5); |
| int g = (int)(green * 255 + 0.5); |
| int b = (int)(blue * 255 + 0.5); |
| parsedColor = new Color(r, g, b); |
| } catch (RuntimeException re) { |
| //wrap in a PropertyException |
| throw new PropertyException(re); |
| } |
| } else { |
| throw new PropertyException("Unknown color format: " + value |
| + ". Must be rgb(r,g,b)"); |
| } |
| return parsedColor; |
| } |
| |
| private static float parseComponent255(String str, String function) |
| throws PropertyException { |
| float component; |
| str = str.trim(); |
| if (str.endsWith("%")) { |
| component = Float.parseFloat(str.substring(0, |
| str.length() - 1)) / 100f; |
| } else { |
| component = Float.parseFloat(str) / 255f; |
| } |
| if ((component < 0.0 || component > 1.0)) { |
| throw new PropertyException("Color value out of range for " + function + ": " |
| + str + ". Valid range: [0..255] or [0%..100%]"); |
| } |
| return component; |
| } |
| |
| private static float parseComponent1(String argument, String function) |
| throws PropertyException { |
| return parseComponent(argument, 0f, 1f, function); |
| } |
| |
| private static float parseComponent(String argument, float min, float max, String function) |
| throws PropertyException { |
| float component = Float.parseFloat(argument.trim()); |
| if ((component < min || component > max)) { |
| throw new PropertyException("Color value out of range for " + function + ": " |
| + argument + ". Valid range: [" + min + ".." + max + "]"); |
| } |
| return component; |
| } |
| |
| private static Color parseFallback(String[] args, String value) throws PropertyException { |
| float red = parseComponent1(args[0], value); |
| float green = parseComponent1(args[1], value); |
| float blue = parseComponent1(args[2], value); |
| //Sun's classlib rounds differently with this constructor than when converting to sRGB |
| //via CIE XYZ. |
| Color sRGB = new Color(red, green, blue); |
| return sRGB; |
| } |
| |
| /** |
| * Parse a color given in the #.... format. |
| * |
| * @param value |
| * the complete line |
| * @return a color if possible |
| * @throws PropertyException |
| * if the format is wrong. |
| */ |
| private static Color parseWithHash(String value) throws PropertyException { |
| Color parsedColor; |
| try { |
| int len = value.length(); |
| int alpha; |
| if (len == 5 || len == 9) { |
| alpha = Integer.parseInt( |
| value.substring((len == 5) ? 3 : 7), 16); |
| } else { |
| alpha = 0xFF; |
| } |
| int red = 0; |
| int green = 0; |
| int blue = 0; |
| if ((len == 4) || (len == 5)) { |
| //multiply by 0x11 = 17 = 255/15 |
| red = Integer.parseInt(value.substring(1, 2), 16) * 0x11; |
| green = Integer.parseInt(value.substring(2, 3), 16) * 0x11; |
| blue = Integer.parseInt(value.substring(3, 4), 16) * 0X11; |
| } else if ((len == 7) || (len == 9)) { |
| red = Integer.parseInt(value.substring(1, 3), 16); |
| green = Integer.parseInt(value.substring(3, 5), 16); |
| blue = Integer.parseInt(value.substring(5, 7), 16); |
| } else { |
| throw new NumberFormatException(); |
| } |
| parsedColor = new Color(red, green, blue, alpha); |
| } catch (RuntimeException re) { |
| throw new PropertyException("Unknown color format: " + value |
| + ". Must be #RGB. #RGBA, #RRGGBB, or #RRGGBBAA"); |
| } |
| return parsedColor; |
| } |
| |
| /** |
| * Parse a color specified using the fop-rgb-icc() function. |
| * |
| * @param value the function call |
| * @return a color if possible |
| * @throws PropertyException if the format is wrong. |
| */ |
| private static Color parseAsFopRgbIcc(FOUserAgent foUserAgent, String value) |
| throws PropertyException { |
| Color parsedColor; |
| int poss = value.indexOf("("); |
| int pose = value.indexOf(")"); |
| if (poss != -1 && pose != -1) { |
| String[] args = value.substring(poss + 1, pose).split(","); |
| |
| try { |
| if (args.length < 5) { |
| throw new PropertyException("Too few arguments for rgb-icc() function"); |
| } |
| |
| //Set up fallback sRGB value |
| Color sRGB = parseFallback(args, value); |
| |
| /* Get and verify ICC profile name */ |
| String iccProfileName = args[3].trim(); |
| if (iccProfileName == null || "".equals(iccProfileName)) { |
| throw new PropertyException("ICC profile name missing"); |
| } |
| ColorSpace colorSpace = null; |
| String iccProfileSrc = null; |
| if (isPseudoProfile(iccProfileName)) { |
| if (CMYK_PSEUDO_PROFILE.equalsIgnoreCase(iccProfileName)) { |
| colorSpace = ColorSpaces.getDeviceCMYKColorSpace(); |
| } else if (SEPARATION_PSEUDO_PROFILE.equalsIgnoreCase(iccProfileName)) { |
| colorSpace = new NamedColorSpace(args[5], sRGB, |
| SEPARATION_PSEUDO_PROFILE, null); |
| } else { |
| assert false : "Incomplete implementation"; |
| } |
| } else { |
| /* Get and verify ICC profile source */ |
| iccProfileSrc = args[4].trim(); |
| if (iccProfileSrc == null || "".equals(iccProfileSrc)) { |
| throw new PropertyException("ICC profile source missing"); |
| } |
| iccProfileSrc = unescapeString(iccProfileSrc); |
| } |
| /* ICC profile arguments */ |
| int componentStart = 4; |
| if (colorSpace instanceof NamedColorSpace) { |
| componentStart++; |
| } |
| float[] iccComponents = new float[args.length - componentStart - 1]; |
| for (int ix = componentStart; ++ix < args.length;) { |
| iccComponents[ix - componentStart - 1] = Float.parseFloat(args[ix].trim()); |
| } |
| if (colorSpace instanceof NamedColorSpace && iccComponents.length == 0) { |
| iccComponents = new float[] {1.0f}; //full tint if not specified |
| } |
| |
| /* Ask FOP factory to get ColorSpace for the specified ICC profile source */ |
| if (foUserAgent != null && iccProfileSrc != null) { |
| RenderingIntent renderingIntent = RenderingIntent.AUTO; |
| //TODO connect to fo:color-profile/@rendering-intent |
| colorSpace = foUserAgent.getColorSpaceCache().get( |
| iccProfileName, iccProfileSrc, renderingIntent); |
| } |
| if (colorSpace != null) { |
| // ColorSpace is available |
| if (ColorSpaces.isDeviceColorSpace(colorSpace)) { |
| //Device-specific colors are handled differently: |
| //sRGB is the primary color with the CMYK as the alternative |
| Color deviceColor = new Color(colorSpace, iccComponents, 1.0f); |
| float[] rgbComps = sRGB.getRGBColorComponents(null); |
| parsedColor = new ColorWithAlternatives( |
| rgbComps[0], rgbComps[1], rgbComps[2], |
| new Color[] {deviceColor}); |
| } else { |
| parsedColor = new ColorWithFallback( |
| colorSpace, iccComponents, 1.0f, null, sRGB); |
| } |
| } else { |
| // ICC profile could not be loaded - use rgb replacement values */ |
| log.warn("Color profile '" + iccProfileSrc |
| + "' not found. Using sRGB replacement values."); |
| parsedColor = sRGB; |
| } |
| } catch (RuntimeException re) { |
| throw new PropertyException(re); |
| } |
| } else { |
| throw new PropertyException("Unknown color format: " + value |
| + ". Must be fop-rgb-icc(r,g,b,NCNAME,src,....)"); |
| } |
| return parsedColor; |
| } |
| |
| /** |
| * Parse a color specified using the fop-rgb-named-color() function. |
| * |
| * @param value the function call |
| * @return a color if possible |
| * @throws PropertyException if the format is wrong. |
| */ |
| private static Color parseAsFopRgbNamedColor(FOUserAgent foUserAgent, String value) |
| throws PropertyException { |
| Color parsedColor; |
| int poss = value.indexOf("("); |
| int pose = value.indexOf(")"); |
| if (poss != -1 && pose != -1) { |
| String[] args = value.substring(poss + 1, pose).split(","); |
| |
| try { |
| if (args.length != 6) { |
| throw new PropertyException("rgb-named-color() function must have 6 arguments"); |
| } |
| |
| //Set up fallback sRGB value |
| Color sRGB = parseFallback(args, value); |
| |
| /* Get and verify ICC profile name */ |
| String iccProfileName = args[3].trim(); |
| if (iccProfileName == null || "".equals(iccProfileName)) { |
| throw new PropertyException("ICC profile name missing"); |
| } |
| ICC_ColorSpace colorSpace = null; |
| String iccProfileSrc; |
| if (isPseudoProfile(iccProfileName)) { |
| throw new IllegalArgumentException( |
| "Pseudo-profiles are not allowed with fop-rgb-named-color()"); |
| } else { |
| /* Get and verify ICC profile source */ |
| iccProfileSrc = args[4].trim(); |
| if (iccProfileSrc == null || "".equals(iccProfileSrc)) { |
| throw new PropertyException("ICC profile source missing"); |
| } |
| iccProfileSrc = unescapeString(iccProfileSrc); |
| } |
| |
| // color name |
| String colorName = unescapeString(args[5].trim()); |
| |
| /* Ask FOP factory to get ColorSpace for the specified ICC profile source */ |
| if (foUserAgent != null && iccProfileSrc != null) { |
| RenderingIntent renderingIntent = RenderingIntent.AUTO; |
| //TODO connect to fo:color-profile/@rendering-intent |
| colorSpace = (ICC_ColorSpace)foUserAgent.getColorSpaceCache().get( |
| iccProfileName, iccProfileSrc, renderingIntent); |
| } |
| if (colorSpace != null) { |
| ICC_Profile profile = colorSpace.getProfile(); |
| if (NamedColorProfileParser.isNamedColorProfile(profile)) { |
| NamedColorProfileParser parser = new NamedColorProfileParser(); |
| NamedColorProfile ncp = parser.parseProfile(profile, |
| iccProfileName, iccProfileSrc); |
| NamedColorSpace ncs = ncp.getNamedColor(colorName); |
| if (ncs != null) { |
| parsedColor = new ColorWithFallback(ncs, |
| new float[] {1.0f}, 1.0f, null, sRGB); |
| } else { |
| log.warn("Color '" + colorName |
| + "' does not exist in named color profile: " + iccProfileSrc); |
| parsedColor = sRGB; |
| } |
| } else { |
| log.warn("ICC profile is no named color profile: " + iccProfileSrc); |
| parsedColor = sRGB; |
| } |
| } else { |
| // ICC profile could not be loaded - use rgb replacement values */ |
| log.warn("Color profile '" + iccProfileSrc |
| + "' not found. Using sRGB replacement values."); |
| parsedColor = sRGB; |
| } |
| } catch (IOException ioe) { |
| //wrap in a PropertyException |
| throw new PropertyException(ioe); |
| } catch (RuntimeException re) { |
| throw new PropertyException(re); |
| //wrap in a PropertyException |
| } |
| } else { |
| throw new PropertyException("Unknown color format: " + value |
| + ". Must be fop-rgb-named-color(r,g,b,NCNAME,src,color-name)"); |
| } |
| return parsedColor; |
| } |
| |
| /** |
| * Parse a color specified using the cie-lab-color() function. |
| * |
| * @param value the function call |
| * @return a color if possible |
| * @throws PropertyException if the format is wrong. |
| */ |
| private static Color parseAsCIELabColor(FOUserAgent foUserAgent, String value) |
| throws PropertyException { |
| Color parsedColor; |
| int poss = value.indexOf("("); |
| int pose = value.indexOf(")"); |
| if (poss != -1 && pose != -1) { |
| try { |
| String[] args = value.substring(poss + 1, pose).split(","); |
| |
| if (args.length != 6) { |
| throw new PropertyException("cie-lab-color() function must have 6 arguments"); |
| } |
| |
| //Set up fallback sRGB value |
| float red = parseComponent255(args[0], value); |
| float green = parseComponent255(args[1], value); |
| float blue = parseComponent255(args[2], value); |
| Color sRGB = new Color(red, green, blue); |
| |
| float l = parseComponent(args[3], 0f, 100f, value); |
| float a = parseComponent(args[4], -127f, 127f, value); |
| float b = parseComponent(args[5], -127f, 127f, value); |
| |
| //Assuming the XSL-FO spec uses the D50 white point |
| CIELabColorSpace cs = ColorSpaces.getCIELabColorSpaceD50(); |
| //use toColor() to have components normalized |
| Color labColor = cs.toColor(l, a, b, 1.0f); |
| //Convert to ColorWithFallback |
| parsedColor = new ColorWithFallback(labColor, sRGB); |
| } catch (RuntimeException re) { |
| throw new PropertyException(re); |
| } |
| } else { |
| throw new PropertyException("Unknown color format: " + value |
| + ". Must be cie-lab-color(r,g,b,Lightness,a-value,b-value)"); |
| } |
| return parsedColor; |
| } |
| |
| private static String unescapeString(String iccProfileSrc) { |
| if (iccProfileSrc.startsWith("\"") || iccProfileSrc.startsWith("'")) { |
| iccProfileSrc = iccProfileSrc.substring(1); |
| } |
| if (iccProfileSrc.endsWith("\"") || iccProfileSrc.endsWith("'")) { |
| iccProfileSrc = iccProfileSrc.substring(0, iccProfileSrc.length() - 1); |
| } |
| return iccProfileSrc; |
| } |
| |
| /** |
| * Parse a color given with the cmyk() function. |
| * |
| * @param value |
| * the complete line |
| * @return a color if possible |
| * @throws PropertyException |
| * if the format is wrong. |
| */ |
| private static Color parseAsCMYK(String value) throws PropertyException { |
| Color parsedColor; |
| int poss = value.indexOf("("); |
| int pose = value.indexOf(")"); |
| if (poss != -1 && pose != -1) { |
| value = value.substring(poss + 1, pose); |
| String[] args = value.split(","); |
| try { |
| if (args.length != 4) { |
| throw new PropertyException( |
| "Invalid number of arguments: cmyk(" + value + ")"); |
| } |
| float cyan = parseComponent1(args[0], value); |
| float magenta = parseComponent1(args[1], value); |
| float yellow = parseComponent1(args[2], value); |
| float black = parseComponent1(args[3], value); |
| float[] comps = new float[] {cyan, magenta, yellow, black}; |
| Color cmykColor = DeviceCMYKColorSpace.createCMYKColor(comps); |
| float[] rgbComps = cmykColor.getRGBColorComponents(null); |
| parsedColor = new ColorWithAlternatives(rgbComps[0], rgbComps[1], rgbComps[2], |
| new Color[] {cmykColor}); |
| } catch (RuntimeException re) { |
| throw new PropertyException(re); |
| } |
| } else { |
| throw new PropertyException("Unknown color format: " + value |
| + ". Must be cmyk(c,m,y,k)"); |
| } |
| return parsedColor; |
| } |
| |
| /** |
| * Creates a re-parsable string representation of the given color. |
| * <p> |
| * First, the color will be converted into the sRGB colorspace. It will then |
| * be printed as #rrggbb, or as #rrrggbbaa if an alpha value is present. |
| * |
| * @param color |
| * the color to represent. |
| * @return a re-parsable string representadion. |
| */ |
| public static String colorToString(Color color) { |
| ColorSpace cs = color.getColorSpace(); |
| if (color instanceof ColorWithAlternatives) { |
| return toFunctionCall((ColorWithAlternatives)color); |
| } else if (cs != null && cs.getType() == ColorSpace.TYPE_CMYK) { |
| StringBuffer sbuf = new StringBuffer(24); |
| float[] cmyk = color.getColorComponents(null); |
| sbuf.append("cmyk(").append(cmyk[0]) |
| .append(",").append(cmyk[1]) |
| .append(",").append(cmyk[2]) |
| .append(",").append(cmyk[3]).append(")"); |
| return sbuf.toString(); |
| } else { |
| return toRGBFunctionCall(color); |
| } |
| } |
| |
| private static String toRGBFunctionCall(Color color) { |
| StringBuffer sbuf = new StringBuffer(); |
| sbuf.append('#'); |
| String s = Integer.toHexString(color.getRed()); |
| if (s.length() == 1) { |
| sbuf.append('0'); |
| } |
| sbuf.append(s); |
| s = Integer.toHexString(color.getGreen()); |
| if (s.length() == 1) { |
| sbuf.append('0'); |
| } |
| sbuf.append(s); |
| s = Integer.toHexString(color.getBlue()); |
| if (s.length() == 1) { |
| sbuf.append('0'); |
| } |
| sbuf.append(s); |
| if (color.getAlpha() != 255) { |
| s = Integer.toHexString(color.getAlpha()); |
| if (s.length() == 1) { |
| sbuf.append('0'); |
| } |
| sbuf.append(s); |
| } |
| return sbuf.toString(); |
| } |
| |
| private static Color getsRGBFallback(ColorWithAlternatives color) { |
| Color fallbackColor; |
| if (color instanceof ColorWithFallback) { |
| fallbackColor = ((ColorWithFallback)color).getFallbackColor(); |
| if (!fallbackColor.getColorSpace().isCS_sRGB()) { |
| fallbackColor = toSRGBColor(fallbackColor); |
| } |
| } else { |
| fallbackColor = toSRGBColor(color); |
| } |
| return fallbackColor; |
| } |
| |
| private static Color toSRGBColor(Color color) { |
| float[] comps; |
| ColorSpace sRGB = ColorSpace.getInstance(ColorSpace.CS_sRGB); |
| comps = color.getRGBColorComponents(null); |
| float[] allComps = color.getComponents(null); |
| float alpha = allComps[allComps.length - 1]; //Alpha is on last component |
| return new Color(sRGB, comps, alpha); |
| } |
| |
| /** |
| * Create string representation of a fop-rgb-icc (or fop-rgb-named-color) function call from |
| * the given color. |
| * @param color the color to turn into a function call |
| * @return the string representing the internal fop-rgb-icc() or fop-rgb-named-color() |
| * function call |
| */ |
| private static String toFunctionCall(ColorWithAlternatives color) { |
| ColorSpace cs = color.getColorSpace(); |
| if (cs.isCS_sRGB() && !color.hasAlternativeColors()) { |
| return toRGBFunctionCall(color); |
| } |
| if (cs instanceof CIELabColorSpace) { |
| return toCIELabFunctionCall(color); |
| } |
| |
| Color specColor = color; |
| if (color.hasAlternativeColors()) { |
| Color alt = color.getAlternativeColors()[0]; |
| if (ColorSpaces.isDeviceColorSpace(alt.getColorSpace())) { |
| cs = alt.getColorSpace(); |
| specColor = alt; |
| } |
| } |
| ColorSpaceOrigin origin = ColorSpaces.getColorSpaceOrigin(cs); |
| String functionName; |
| |
| Color fallbackColor = getsRGBFallback(color); |
| float[] rgb = fallbackColor.getColorComponents(null); |
| assert rgb.length == 3; |
| StringBuffer sb = new StringBuffer(40); |
| sb.append("("); |
| sb.append(rgb[0]).append(","); |
| sb.append(rgb[1]).append(","); |
| sb.append(rgb[2]).append(","); |
| String profileName = origin.getProfileName(); |
| sb.append(profileName).append(","); |
| if (origin.getProfileURI() != null) { |
| sb.append("\"").append(origin.getProfileURI()).append("\""); |
| } |
| |
| if (cs instanceof NamedColorSpace) { |
| NamedColorSpace ncs = (NamedColorSpace)cs; |
| if (SEPARATION_PSEUDO_PROFILE.equalsIgnoreCase(profileName)) { |
| functionName = "fop-rgb-icc"; |
| } else { |
| functionName = "fop-rgb-named-color"; |
| } |
| sb.append(",").append(ncs.getColorName()); |
| } else { |
| functionName = "fop-rgb-icc"; |
| float[] colorComponents = specColor.getColorComponents(null); |
| for (float colorComponent : colorComponents) { |
| sb.append(","); |
| sb.append(colorComponent); |
| } |
| } |
| sb.append(")"); |
| return functionName + sb.toString(); |
| } |
| |
| private static String toCIELabFunctionCall(ColorWithAlternatives color) { |
| Color fallbackColor = getsRGBFallback(color); |
| StringBuffer sb = new StringBuffer("cie-lab-color("); |
| sb.append(fallbackColor.getRed()).append(','); |
| sb.append(fallbackColor.getGreen()).append(','); |
| sb.append(fallbackColor.getBlue()); |
| CIELabColorSpace cs = (CIELabColorSpace)color.getColorSpace(); |
| float[] lab = cs.toNativeComponents(color.getColorComponents(null)); |
| for (int i = 0; i < 3; i++) { |
| sb.append(',').append(lab[i]); |
| } |
| sb.append(')'); |
| return sb.toString(); |
| } |
| |
| private static Color createColor(int r, int g, int b) { |
| return new Color(r, g, b); |
| } |
| |
| /** |
| * Initializes the colorMap with some predefined values. |
| */ |
| private static void initializeColorMap() { |
| colorMap = Collections.synchronizedMap(new java.util.HashMap<String, Color>()); |
| |
| colorMap.put("aliceblue", createColor(240, 248, 255)); |
| colorMap.put("antiquewhite", createColor(250, 235, 215)); |
| colorMap.put("aqua", createColor(0, 255, 255)); |
| colorMap.put("aquamarine", createColor(127, 255, 212)); |
| colorMap.put("azure", createColor(240, 255, 255)); |
| colorMap.put("beige", createColor(245, 245, 220)); |
| colorMap.put("bisque", createColor(255, 228, 196)); |
| colorMap.put("black", createColor(0, 0, 0)); |
| colorMap.put("blanchedalmond", createColor(255, 235, 205)); |
| colorMap.put("blue", createColor(0, 0, 255)); |
| colorMap.put("blueviolet", createColor(138, 43, 226)); |
| colorMap.put("brown", createColor(165, 42, 42)); |
| colorMap.put("burlywood", createColor(222, 184, 135)); |
| colorMap.put("cadetblue", createColor(95, 158, 160)); |
| colorMap.put("chartreuse", createColor(127, 255, 0)); |
| colorMap.put("chocolate", createColor(210, 105, 30)); |
| colorMap.put("coral", createColor(255, 127, 80)); |
| colorMap.put("cornflowerblue", createColor(100, 149, 237)); |
| colorMap.put("cornsilk", createColor(255, 248, 220)); |
| colorMap.put("crimson", createColor(220, 20, 60)); |
| colorMap.put("cyan", createColor(0, 255, 255)); |
| colorMap.put("darkblue", createColor(0, 0, 139)); |
| colorMap.put("darkcyan", createColor(0, 139, 139)); |
| colorMap.put("darkgoldenrod", createColor(184, 134, 11)); |
| colorMap.put("darkgray", createColor(169, 169, 169)); |
| colorMap.put("darkgreen", createColor(0, 100, 0)); |
| colorMap.put("darkgrey", createColor(169, 169, 169)); |
| colorMap.put("darkkhaki", createColor(189, 183, 107)); |
| colorMap.put("darkmagenta", createColor(139, 0, 139)); |
| colorMap.put("darkolivegreen", createColor(85, 107, 47)); |
| colorMap.put("darkorange", createColor(255, 140, 0)); |
| colorMap.put("darkorchid", createColor(153, 50, 204)); |
| colorMap.put("darkred", createColor(139, 0, 0)); |
| colorMap.put("darksalmon", createColor(233, 150, 122)); |
| colorMap.put("darkseagreen", createColor(143, 188, 143)); |
| colorMap.put("darkslateblue", createColor(72, 61, 139)); |
| colorMap.put("darkslategray", createColor(47, 79, 79)); |
| colorMap.put("darkslategrey", createColor(47, 79, 79)); |
| colorMap.put("darkturquoise", createColor(0, 206, 209)); |
| colorMap.put("darkviolet", createColor(148, 0, 211)); |
| colorMap.put("deeppink", createColor(255, 20, 147)); |
| colorMap.put("deepskyblue", createColor(0, 191, 255)); |
| colorMap.put("dimgray", createColor(105, 105, 105)); |
| colorMap.put("dimgrey", createColor(105, 105, 105)); |
| colorMap.put("dodgerblue", createColor(30, 144, 255)); |
| colorMap.put("firebrick", createColor(178, 34, 34)); |
| colorMap.put("floralwhite", createColor(255, 250, 240)); |
| colorMap.put("forestgreen", createColor(34, 139, 34)); |
| colorMap.put("fuchsia", createColor(255, 0, 255)); |
| colorMap.put("gainsboro", createColor(220, 220, 220)); |
| colorMap.put("ghostwhite", createColor(248, 248, 255)); |
| colorMap.put("gold", createColor(255, 215, 0)); |
| colorMap.put("goldenrod", createColor(218, 165, 32)); |
| colorMap.put("gray", createColor(128, 128, 128)); |
| colorMap.put("green", createColor(0, 128, 0)); |
| colorMap.put("greenyellow", createColor(173, 255, 47)); |
| colorMap.put("grey", createColor(128, 128, 128)); |
| colorMap.put("honeydew", createColor(240, 255, 240)); |
| colorMap.put("hotpink", createColor(255, 105, 180)); |
| colorMap.put("indianred", createColor(205, 92, 92)); |
| colorMap.put("indigo", createColor(75, 0, 130)); |
| colorMap.put("ivory", createColor(255, 255, 240)); |
| colorMap.put("khaki", createColor(240, 230, 140)); |
| colorMap.put("lavender", createColor(230, 230, 250)); |
| colorMap.put("lavenderblush", createColor(255, 240, 245)); |
| colorMap.put("lawngreen", createColor(124, 252, 0)); |
| colorMap.put("lemonchiffon", createColor(255, 250, 205)); |
| colorMap.put("lightblue", createColor(173, 216, 230)); |
| colorMap.put("lightcoral", createColor(240, 128, 128)); |
| colorMap.put("lightcyan", createColor(224, 255, 255)); |
| colorMap.put("lightgoldenrodyellow", createColor(250, 250, 210)); |
| colorMap.put("lightgray", createColor(211, 211, 211)); |
| colorMap.put("lightgreen", createColor(144, 238, 144)); |
| colorMap.put("lightgrey", createColor(211, 211, 211)); |
| colorMap.put("lightpink", createColor(255, 182, 193)); |
| colorMap.put("lightsalmon", createColor(255, 160, 122)); |
| colorMap.put("lightseagreen", createColor(32, 178, 170)); |
| colorMap.put("lightskyblue", createColor(135, 206, 250)); |
| colorMap.put("lightslategray", createColor(119, 136, 153)); |
| colorMap.put("lightslategrey", createColor(119, 136, 153)); |
| colorMap.put("lightsteelblue", createColor(176, 196, 222)); |
| colorMap.put("lightyellow", createColor(255, 255, 224)); |
| colorMap.put("lime", createColor(0, 255, 0)); |
| colorMap.put("limegreen", createColor(50, 205, 50)); |
| colorMap.put("linen", createColor(250, 240, 230)); |
| colorMap.put("magenta", createColor(255, 0, 255)); |
| colorMap.put("maroon", createColor(128, 0, 0)); |
| colorMap.put("mediumaquamarine", createColor(102, 205, 170)); |
| colorMap.put("mediumblue", createColor(0, 0, 205)); |
| colorMap.put("mediumorchid", createColor(186, 85, 211)); |
| colorMap.put("mediumpurple", createColor(147, 112, 219)); |
| colorMap.put("mediumseagreen", createColor(60, 179, 113)); |
| colorMap.put("mediumslateblue", createColor(123, 104, 238)); |
| colorMap.put("mediumspringgreen", createColor(0, 250, 154)); |
| colorMap.put("mediumturquoise", createColor(72, 209, 204)); |
| colorMap.put("mediumvioletred", createColor(199, 21, 133)); |
| colorMap.put("midnightblue", createColor(25, 25, 112)); |
| colorMap.put("mintcream", createColor(245, 255, 250)); |
| colorMap.put("mistyrose", createColor(255, 228, 225)); |
| colorMap.put("moccasin", createColor(255, 228, 181)); |
| colorMap.put("navajowhite", createColor(255, 222, 173)); |
| colorMap.put("navy", createColor(0, 0, 128)); |
| colorMap.put("oldlace", createColor(253, 245, 230)); |
| colorMap.put("olive", createColor(128, 128, 0)); |
| colorMap.put("olivedrab", createColor(107, 142, 35)); |
| colorMap.put("orange", createColor(255, 165, 0)); |
| colorMap.put("orangered", createColor(255, 69, 0)); |
| colorMap.put("orchid", createColor(218, 112, 214)); |
| colorMap.put("palegoldenrod", createColor(238, 232, 170)); |
| colorMap.put("palegreen", createColor(152, 251, 152)); |
| colorMap.put("paleturquoise", createColor(175, 238, 238)); |
| colorMap.put("palevioletred", createColor(219, 112, 147)); |
| colorMap.put("papayawhip", createColor(255, 239, 213)); |
| colorMap.put("peachpuff", createColor(255, 218, 185)); |
| colorMap.put("peru", createColor(205, 133, 63)); |
| colorMap.put("pink", createColor(255, 192, 203)); |
| colorMap.put("plum ", createColor(221, 160, 221)); |
| colorMap.put("plum", createColor(221, 160, 221)); |
| colorMap.put("powderblue", createColor(176, 224, 230)); |
| colorMap.put("purple", createColor(128, 0, 128)); |
| colorMap.put("red", createColor(255, 0, 0)); |
| colorMap.put("rosybrown", createColor(188, 143, 143)); |
| colorMap.put("royalblue", createColor(65, 105, 225)); |
| colorMap.put("saddlebrown", createColor(139, 69, 19)); |
| colorMap.put("salmon", createColor(250, 128, 114)); |
| colorMap.put("sandybrown", createColor(244, 164, 96)); |
| colorMap.put("seagreen", createColor(46, 139, 87)); |
| colorMap.put("seashell", createColor(255, 245, 238)); |
| colorMap.put("sienna", createColor(160, 82, 45)); |
| colorMap.put("silver", createColor(192, 192, 192)); |
| colorMap.put("skyblue", createColor(135, 206, 235)); |
| colorMap.put("slateblue", createColor(106, 90, 205)); |
| colorMap.put("slategray", createColor(112, 128, 144)); |
| colorMap.put("slategrey", createColor(112, 128, 144)); |
| colorMap.put("snow", createColor(255, 250, 250)); |
| colorMap.put("springgreen", createColor(0, 255, 127)); |
| colorMap.put("steelblue", createColor(70, 130, 180)); |
| colorMap.put("tan", createColor(210, 180, 140)); |
| colorMap.put("teal", createColor(0, 128, 128)); |
| colorMap.put("thistle", createColor(216, 191, 216)); |
| colorMap.put("tomato", createColor(255, 99, 71)); |
| colorMap.put("turquoise", createColor(64, 224, 208)); |
| colorMap.put("violet", createColor(238, 130, 238)); |
| colorMap.put("wheat", createColor(245, 222, 179)); |
| colorMap.put("white", createColor(255, 255, 255)); |
| colorMap.put("whitesmoke", createColor(245, 245, 245)); |
| colorMap.put("yellow", createColor(255, 255, 0)); |
| colorMap.put("yellowgreen", createColor(154, 205, 50)); |
| colorMap.put("transparent", new ColorWithAlternatives(0, 0, 0, 0, null)); |
| } |
| |
| /** |
| * Lightens up a color for groove, ridge, inset and outset border effects. |
| * @param col the color to lighten up |
| * @param factor factor by which to lighten up (negative values darken the color) |
| * @return the modified color |
| */ |
| public static Color lightenColor(Color col, float factor) { |
| return org.apache.xmlgraphics.java2d.color.ColorUtil.lightenColor(col, factor); |
| } |
| |
| /** |
| * Indicates whether the given color profile name is one of the pseudo-profiles supported |
| * by FOP (ex. #CMYK). |
| * @param colorProfileName the color profile name to check |
| * @return true if the color profile name is of a built-in pseudo-profile |
| */ |
| public static boolean isPseudoProfile(String colorProfileName) { |
| return CMYK_PSEUDO_PROFILE.equalsIgnoreCase(colorProfileName) |
| || SEPARATION_PSEUDO_PROFILE.equalsIgnoreCase(colorProfileName); |
| } |
| |
| /** |
| * Indicates whether the color is a gray value. |
| * @param col the color |
| * @return true if it is a gray value |
| */ |
| public static boolean isGray(Color col) { |
| return org.apache.xmlgraphics.java2d.color.ColorUtil.isGray(col); |
| } |
| |
| /** |
| * Creates an uncalibrated CMYK color with the given gray value. |
| * @param black the gray component (0 - 1) |
| * @return the CMYK color |
| */ |
| public static Color toCMYKGrayColor(float black) { |
| return org.apache.xmlgraphics.java2d.color.ColorUtil.toCMYKGrayColor(black); |
| } |
| } |