/*
 * 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: PDFColor.java 1758773 2016-09-01 13:02:29Z ssteiner $ */

package org.apache.fop.pdf;

import java.awt.color.ColorSpace;
import java.util.ArrayList;
import java.util.List;

import org.apache.xmlgraphics.java2d.color.DeviceCMYKColorSpace;

PDF Color object. It is currently only used to hold the transparent color of a masked bitmap image. And in this context, only RGB and Gray values are used.

Use of this class is discouraged. PDFColorHandler is now used for in-content color selection. For masked bitmaps, it may be wiser to switch to Color in the long run.

/** * PDF Color object. It is currently only used to hold the transparent color of a masked bitmap * image. And in this context, only RGB and Gray values are used. * <p> * Use of this class is discouraged. {@link PDFColorHandler} is now used for in-content color * selection. For masked bitmaps, it may be wiser to switch to {@link java.awt.Color} in the long run. */
public class PDFColor extends PDFPathPaint { // could be 3.0 as well. private static double blackFactor = 2.0; private double red = -1.0; private double green = -1.0; private double blue = -1.0; private double cyan = -1.0; private double magenta = -1.0; private double yellow = -1.0; private double black = -1.0;
Create a PDF color with double values ranging from 0 to 1.
Params:
  • theRed – the red double value
  • theGreen – the green double value
  • theBlue – the blue double value
/** * Create a PDF color with double values ranging from 0 to 1. * * @param theRed the red double value * @param theGreen the green double value * @param theBlue the blue double value */
public PDFColor(double theRed, double theGreen, double theBlue) { // super(theNumber); this.colorSpace = new PDFDeviceColorSpace(PDFDeviceColorSpace.DEVICE_RGB); this.red = theRed; this.green = theGreen; this.blue = theBlue; }
Create a PDF color from a java.awt.Color object. Different Color objects are handled differently. Cases recognized are. 1. CMYK color 3. 'Normal' java.awt.Color (RGB case assumed or implicit conversion to sRGB)
Params:
  • col – the java.awt.Color object for which to create a PDFColor object
/** * Create a PDF color from a java.awt.Color object. * * Different Color objects are handled differently. Cases recognized are. * * 1. CMYK color * 3. 'Normal' java.awt.Color (RGB case assumed or implicit conversion to sRGB) * * @param col the java.awt.Color object for which to create a PDFColor object */
public PDFColor(java.awt.Color col) { ColorSpace cs = col.getColorSpace(); if (cs != null && cs instanceof DeviceCMYKColorSpace) { // CMYK case this.colorSpace = new PDFDeviceColorSpace(PDFDeviceColorSpace.DEVICE_CMYK); float[] cmyk = col.getColorComponents(null); this.cyan = cmyk[0]; this.magenta = cmyk[1]; this.yellow = cmyk[2]; this.black = cmyk[3]; } else { // Default (RGB) Color (ICC Colors are converted to sRGB, too) this.colorSpace = new PDFDeviceColorSpace(PDFDeviceColorSpace.DEVICE_RGB); float[] comps = new float[3]; comps = col.getColorComponents(comps); this.red = comps[0]; this.green = comps[1]; this.blue = comps[2]; } }
Create a PDF color with int values ranging from 0 to 255
Params:
  • theRed – the red integer value
  • theGreen – the green integer value
  • theBlue – the blue integer value
/** * Create a PDF color with int values ranging from 0 to 255 * * @param theRed the red integer value * @param theGreen the green integer value * @param theBlue the blue integer value */
public PDFColor(int theRed, int theGreen, int theBlue) { this(((double)theRed) / 255d, ((double)theGreen) / 255d, ((double)theBlue) / 255d); }
Create a PDF color with CMYK values.
Params:
  • theCyan – the cyan value
  • theMagenta – the magenta value
  • theYellow – the yellow value
  • theBlack – the black value
/** * Create a PDF color with CMYK values. * * @param theCyan the cyan value * @param theMagenta the magenta value * @param theYellow the yellow value * @param theBlack the black value */
public PDFColor(double theCyan, double theMagenta, double theYellow, double theBlack) { this.colorSpace = new PDFDeviceColorSpace(PDFDeviceColorSpace.DEVICE_CMYK); this.cyan = theCyan; this.magenta = theMagenta; this.yellow = theYellow; this.black = theBlack; }
Return a vector representation of the color in the appropriate colorspace.
Returns:a list containing the Double values of the color
/** * Return a vector representation of the color * in the appropriate colorspace. * * @return a list containing the Double values of the color */
public List getVector() { List theColorVector = new ArrayList(); if (this.colorSpace.getColorSpace() == PDFDeviceColorSpace.DEVICE_RGB) { // RGB theColorVector.add(this.red); theColorVector.add(this.green); theColorVector.add(this.blue); } else if (this.colorSpace.getColorSpace() == PDFDeviceColorSpace.DEVICE_CMYK) { // CMYK theColorVector.add(this.cyan); theColorVector.add(this.magenta); theColorVector.add(this.yellow); theColorVector.add(this.black); } else { // GRAY theColorVector.add(this.black); } return (theColorVector); }
Get the red component.
Returns:the red double value
/** * Get the red component. * * @return the red double value */
public double red() { return (this.red); }
Get the green component.
Returns:the green double value
/** * Get the green component. * * @return the green double value */
public double green() { return (this.green); }
Get the blue component.
Returns:the blue double value
/** * Get the blue component. * * @return the blue double value */
public double blue() { return (this.blue); }
Get the red integer component.
Returns:the red integer value
/** * Get the red integer component. * * @return the red integer value */
public int red255() { return (int)(this.red * 255d); }
Get the green integer component.
Returns:the green integer value
/** * Get the green integer component. * * @return the green integer value */
public int green255() { return (int)(this.green * 255d); }
Get the blue integer component.
Returns:the blue integer value
/** * Get the blue integer component. * * @return the blue integer value */
public int blue255() { return (int)(this.blue * 255d); }
Get the cyan component.
Returns:the cyan double value
/** * Get the cyan component. * * @return the cyan double value */
public double cyan() { return (this.cyan); }
Get the magenta component.
Returns:the magenta double value
/** * Get the magenta component. * * @return the magenta double value */
public double magenta() { return (this.magenta); }
Get the yellow component.
Returns:the yellow double value
/** * Get the yellow component. * * @return the yellow double value */
public double yellow() { return (this.yellow); }
Get the black component.
Returns:the black double value
/** * Get the black component. * * @return the black double value */
public double black() { return (this.black); }
Set the color space for this color. If the new color space is different the values are converted to the new color space.
Params:
  • theColorSpace – the new color space
/** * Set the color space for this color. * If the new color space is different the values are converted * to the new color space. * * @param theColorSpace the new color space */
public void setColorSpace(int theColorSpace) { int theOldColorSpace = this.colorSpace.getColorSpace(); if (theOldColorSpace != theColorSpace) { if (theOldColorSpace == PDFDeviceColorSpace.DEVICE_RGB) { if (theColorSpace == PDFDeviceColorSpace.DEVICE_CMYK) { this.convertRGBtoCMYK(); } else { // convert to Gray? this.convertRGBtoGRAY(); } } else if (theOldColorSpace == PDFDeviceColorSpace.DEVICE_CMYK) { if (theColorSpace == PDFDeviceColorSpace.DEVICE_RGB) { this.convertCMYKtoRGB(); } else { // convert to Gray? this.convertCMYKtoGRAY(); } } else { // used to be Gray if (theColorSpace == PDFDeviceColorSpace.DEVICE_RGB) { this.convertGRAYtoRGB(); } else { // convert to CMYK? this.convertGRAYtoCMYK(); } } this.colorSpace.setColorSpace(theColorSpace); } }
Get the PDF output string for this color. This returns the string to be inserted into PDF for setting the current color.
Params:
  • fillNotStroke – whether to return fill or stroke command
Returns:the PDF string for setting the fill/stroke color
/** * Get the PDF output string for this color. * This returns the string to be inserted into PDF for setting * the current color. * * @param fillNotStroke whether to return fill or stroke command * @return the PDF string for setting the fill/stroke color */
public String getColorSpaceOut(boolean fillNotStroke) { StringBuffer p = new StringBuffer(""); if (this.colorSpace.getColorSpace() == PDFDeviceColorSpace.DEVICE_RGB) { // colorspace is RGB // according to pdfspec 12.1 p.399 // if the colors are the same then just use the g or G operator boolean same = false; if (this.red == this.green && this.red == this.blue) { same = true; } // output RGB if (fillNotStroke) { // fill if (same) { p.append(PDFNumber.doubleOut(this.red) + " g\n"); } else { p.append(PDFNumber.doubleOut(this.red) + " " + PDFNumber.doubleOut(this.green) + " " + PDFNumber.doubleOut(this.blue) + " rg\n"); } } else { // stroke/border if (same) { p.append(PDFNumber.doubleOut(this.red) + " G\n"); } else { p.append(PDFNumber.doubleOut(this.red) + " " + PDFNumber.doubleOut(this.green) + " " + PDFNumber.doubleOut(this.blue) + " RG\n"); } } } else if (this.colorSpace.getColorSpace() == PDFDeviceColorSpace.DEVICE_CMYK) { // colorspace is CMYK if (fillNotStroke) { // fill p.append(PDFNumber.doubleOut(this.cyan) + " " + PDFNumber.doubleOut(this.magenta) + " " + PDFNumber.doubleOut(this.yellow) + " " + PDFNumber.doubleOut(this.black) + " k\n"); } else { // stroke p.append(PDFNumber.doubleOut(this.cyan) + " " + PDFNumber.doubleOut(this.magenta) + " " + PDFNumber.doubleOut(this.yellow) + " " + PDFNumber.doubleOut(this.black) + " K\n"); } } else { // means we're in DeviceGray or Unknown. // assume we're in DeviceGray, because otherwise we're screwed. if (fillNotStroke) { p.append(PDFNumber.doubleOut(this.black) + " g\n"); } else { p.append(PDFNumber.doubleOut(this.black) + " G\n"); } } return (p.toString()); }
Convert the color from CMYK to RGB.
/** * Convert the color from CMYK to RGB. */
protected void convertCMYKtoRGB() { // convert CMYK to RGB this.red = 1.0 - this.cyan; this.green = 1.0 - this.green; this.blue = 1.0 - this.yellow; this.red = (this.black / PDFColor.blackFactor) + this.red; this.green = (this.black / PDFColor.blackFactor) + this.green; this.blue = (this.black / PDFColor.blackFactor) + this.blue; }
Convert the color from RGB to CMYK.
/** * Convert the color from RGB to CMYK. */
protected void convertRGBtoCMYK() { // convert RGB to CMYK this.cyan = 1.0 - this.red; this.magenta = 1.0 - this.green; this.yellow = 1.0 - this.blue; this.black = 0.0; /* * If you want to calculate black, uncomment this * //pick the lowest color * tempDouble = this.red; * * if (this.green < tempDouble) * tempDouble = this.green; * * if (this.blue < tempDouble) * tempDouble = this.blue; * * this.black = tempDouble / this.blackFactor; */ }
Convert the color from Gray to RGB.
/** * Convert the color from Gray to RGB. */
protected void convertGRAYtoRGB() { this.red = 1.0 - this.black; this.green = 1.0 - this.black; this.blue = 1.0 - this.black; }
Convert the color from Gray to CMYK.
/** * Convert the color from Gray to CMYK. */
protected void convertGRAYtoCMYK() { this.cyan = this.black; this.magenta = this.black; this.yellow = this.black; // this.black=0.0;//? }
Convert the color from CMYK to Gray.
/** * Convert the color from CMYK to Gray. */
protected void convertCMYKtoGRAY() { double tempDouble = 0.0; // pick the lowest color tempDouble = this.cyan; if (this.magenta < tempDouble) { tempDouble = this.magenta; } if (this.yellow < tempDouble) { tempDouble = this.yellow; } this.black = (tempDouble / PDFColor.blackFactor); }
Convert the color from RGB to Gray.
/** * Convert the color from RGB to Gray. */
protected void convertRGBtoGRAY() { double tempDouble = 0.0; // pick the lowest color tempDouble = this.red; if (this.green < tempDouble) { tempDouble = this.green; } if (this.blue < tempDouble) { tempDouble = this.blue; } this.black = 1.0 - (tempDouble / PDFColor.blackFactor); }
Create pdf. Not used for this object.
Returns:the bytes for the pdf
/** * Create pdf. * Not used for this object. * * @return the bytes for the pdf */
public byte[] toPDF() { return (new byte[0]); }
{@inheritDoc}
/** {@inheritDoc} */
protected boolean contentEquals(PDFObject obj) { if (!(obj instanceof PDFColor)) { return false; } PDFColor color = (PDFColor)obj; if (color.red == this.red && color.green == this.green && color.blue == this.blue) { return true; } return false; } }