org.apache.fop.util.bitmap.JAIMonochromeBitmapConverter Maven / Gradle / Ivy
Show all versions of org.apache.fop Show documentation
/*
* 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: JAIMonochromeBitmapConverter.java 1610839 2014-07-15 20:25:58Z vhennebert $ */
package org.apache.fop.util.bitmap;
import java.awt.RenderingHints;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.awt.image.DataBuffer;
import java.awt.image.IndexColorModel;
import java.awt.image.RenderedImage;
import java.awt.image.renderable.ParameterBlock;
import javax.media.jai.ColorCube;
import javax.media.jai.ImageLayout;
import javax.media.jai.JAI;
import javax.media.jai.KernelJAI;
import javax.media.jai.LookupTableJAI;
import javax.media.jai.PlanarImage;
/**
* Implementation of the MonochromeBitmapConverter which uses Java Advanced Imaging (JAI)
* to convert grayscale bitmaps to monochrome bitmaps. JAI provides better dithering options
* including error diffusion dithering.
*
* If you call setHint("quality", "true") on the instance you can enabled error diffusion
* dithering which produces a nicer result but is also a lot slower.
*/
public class JAIMonochromeBitmapConverter implements
MonochromeBitmapConverter {
private boolean isErrorDiffusion;
/** {@inheritDoc} */
public void setHint(String name, String value) {
if ("quality".equalsIgnoreCase(name)) {
isErrorDiffusion = "true".equalsIgnoreCase(value);
}
}
/** {@inheritDoc} */
public RenderedImage convertToMonochrome(BufferedImage img) {
return convertToMonochromePlanarImage(img);
}
private PlanarImage convertToMonochromePlanarImage(BufferedImage img) {
if (img.getColorModel().getColorSpace().getNumComponents() != 1) {
img = BitmapImageUtil.convertToGrayscale(img, null);
}
// Load the ParameterBlock for the dithering operation
// and set the operation name.
ParameterBlock pb = new ParameterBlock();
pb.addSource(img);
String opName = null;
if (isErrorDiffusion) {
opName = "errordiffusion";
LookupTableJAI lut = new LookupTableJAI(new byte[] {(byte)0x00, (byte)0xff});
pb.add(lut);
pb.add(KernelJAI.ERROR_FILTER_FLOYD_STEINBERG);
} else {
opName = "ordereddither";
//Create the color cube.
ColorCube colorMap = ColorCube.createColorCube(DataBuffer.TYPE_BYTE,
0, new int[] {2});
pb.add(colorMap);
pb.add(KernelJAI.DITHER_MASK_441);
}
//Create an image layout for a monochrome b/w image
ImageLayout layout = new ImageLayout();
byte[] map = new byte[] {(byte)0x00, (byte)0xff};
ColorModel cm = new IndexColorModel(1, 2, map, map, map);
layout.setColorModel(cm);
// Create a hint containing the layout.
RenderingHints hints = new RenderingHints(JAI.KEY_IMAGE_LAYOUT, layout);
// Dither the image.
PlanarImage dst = JAI.create(opName, pb, hints);
//Convert it to a BufferedImage
return dst;
}
}