boofcv.io.image.ConvertBufferedImage Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of boofcv-io Show documentation
Show all versions of boofcv-io Show documentation
BoofCV is an open source Java library for real-time computer vision and robotics applications.
/*
* Copyright (c) 2011-2017, Peter Abeles. All Rights Reserved.
*
* This file is part of BoofCV (http://boofcv.org).
*
* Licensed 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.
*/
package boofcv.io.image;
import boofcv.core.image.GeneralizedImageOps;
import boofcv.struct.image.*;
import javax.swing.*;
import java.awt.*;
import java.awt.color.ColorSpace;
import java.awt.image.*;
import java.lang.reflect.Array;
/**
* Functions for converting to and from {@link BufferedImage}.
*
* @author Peter Abeles
*/
@SuppressWarnings({"Duplicates", "unchecked"})
public class ConvertBufferedImage {
/**
* If the provided image does not have the same shape and same type a new one is declared and returned.
*/
public static BufferedImage checkDeclare( int width , int height , BufferedImage image , int type ) {
if( image == null )
return new BufferedImage(width,height,type);
if( image.getType() != type )
return new BufferedImage(width,height,type);
if( image.getWidth() != width || image.getHeight() != height )
return new BufferedImage(width,height,type);
return image;
}
public static BufferedImage checkDeclare( BufferedImage template , BufferedImage target ) {
int width = template.getWidth();
int height = template.getHeight();
int type = template.getType();
if( type == 0 ) {
if( target != null )
type = target.getType();
else
type = BufferedImage.TYPE_INT_RGB;
}
if( target == null )
return new BufferedImage(width,height,type);
if( target.getType() != type )
return new BufferedImage(width,height,type);
if( target.getWidth() != width || target.getHeight() != height )
return new BufferedImage(width,height,type);
return target;
}
/**
* Copies the original image into the output image. If it can't do a copy a new image is created and returned
* @param original Original image
* @param output (Optional) Storage for copy.
* @return The copied image. May be a new instance
*/
public static BufferedImage checkCopy( BufferedImage original , BufferedImage output ) {
ColorModel cm = original.getColorModel();
boolean isAlphaPremultiplied = cm.isAlphaPremultiplied();
if( output == null || original.getWidth() != output.getWidth() || original.getHeight() != output.getHeight() ||
original.getType() != output.getType() ) {
WritableRaster raster = original.copyData(original.getRaster().createCompatibleWritableRaster());
return new BufferedImage(cm, raster, isAlphaPremultiplied, null);
}
original.copyData(output.getRaster());
return output;
}
/**
* Returns an image which doesn't have an alpha channel. If the input image doesn't have an alpha
* channel to start then its returned as is. Otherwise a new image is created and the RGB channels are
* copied and the new image returned.
*
* @param image Input image
* @return Image without an alpha channel
*/
public static BufferedImage stripAlphaChannel( BufferedImage image ) {
int numBands = image.getRaster().getNumBands();
if( numBands == 4 ) {
BufferedImage output = new BufferedImage(image.getWidth(),image.getHeight(),BufferedImage.TYPE_INT_RGB);
output.createGraphics().drawImage(image,0,0,null);
return output;
} else {
return image;
}
}
/**
* For BufferedImage stored as a byte array internally it extracts an
* interleaved image. The input image and the returned image will both
* share the same internal data array. Using this function allows unnecessary
* memory copying to be avoided.
*
* @param img Image whose internal data is extracted and wrapped.
* @return An image whose internal data is the same as the input image.
*/
public static InterleavedU8 extractInterleavedU8(BufferedImage img) {
DataBuffer buffer = img.getRaster().getDataBuffer();
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(img) ) {
WritableRaster raster = img.getRaster();
InterleavedU8 ret = new InterleavedU8();
ret.width = img.getWidth();
ret.height = img.getHeight();
ret.startIndex = ConvertRaster.getOffset(raster);
ret.imageType.numBands = raster.getNumBands();
ret.numBands = raster.getNumBands();
ret.stride = ConvertRaster.stride(raster);
ret.data = ((DataBufferByte)buffer).getData();
ret.subImage = ret.startIndex != 0;
return ret;
}
throw new IllegalArgumentException("Buffered image does not have an interleaved byte raster");
}
/**
* For BufferedImage stored as a byte array internally it extracts an
* image. The input image and the returned image will both
* share the same internal data array. Using this function allows unnecessary
* memory copying to be avoided.
*
* @param img Image whose internal data is extracted and wrapped.
* @return An image whose internal data is the same as the input image.
*/
public static GrayU8 extractGrayU8(BufferedImage img) {
WritableRaster raster = img.getRaster();
DataBuffer buffer = raster.getDataBuffer();
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(img) ) {
if (raster.getNumBands() != 1)
throw new IllegalArgumentException("Input image has more than one channel");
GrayU8 ret = new GrayU8();
ret.width = img.getWidth();
ret.height = img.getHeight();
ret.startIndex = ConvertRaster.getOffset(img.getRaster());
ret.stride = ConvertRaster.stride(img.getRaster());
ret.data = ((DataBufferByte)buffer).getData();
return ret;
}
throw new IllegalArgumentException("Buffered image does not have a gray scale byte raster");
}
/**
* Creates a new BufferedImage that internally uses the same data as the provided
* {@link InterleavedU8}. If 3 bands then the image will be of type TYPE_3BYTE_BGR
* or if 1 band TYPE_BYTE_GRAY.
*
* @param img Input image who's data will be wrapped by the returned BufferedImage.
* @return BufferedImage which shared data with the input image.
*/
public static BufferedImage extractBuffered(InterleavedU8 img) {
if (img.isSubimage())
throw new IllegalArgumentException("Sub-images are not supported for this operation");
final int width = img.width;
final int height = img.height;
final int numBands = img.numBands;
// wrap the byte array
DataBuffer bufferByte = new DataBufferByte(img.data, width * height * numBands, 0);
ColorModel colorModel;
int[] bOffs = null;
if (numBands == 3) {
ColorSpace cs = ColorSpace.getInstance(ColorSpace.CS_sRGB);
int[] nBits = {8, 8, 8};
bOffs = new int[]{2, 1, 0};
colorModel = new ComponentColorModel(cs, nBits, false, false,
Transparency.OPAQUE,
DataBuffer.TYPE_BYTE);
} else if (numBands == 1) {
ColorSpace cs = ColorSpace.getInstance(ColorSpace.CS_GRAY);
int[] nBits = {8};
bOffs = new int[]{0};
colorModel = new ComponentColorModel(cs, nBits, false, true,
Transparency.OPAQUE,
DataBuffer.TYPE_BYTE);
} else {
throw new IllegalArgumentException("Only 1 or 3 bands supported");
}
// Create a raster using the sample model and data buffer
WritableRaster raster = Raster.createInterleavedRaster(
bufferByte, width, height, img.stride, numBands, bOffs, new Point(0, 0));
// Combine the color model and raster into a buffered image
return new BufferedImage(colorModel, raster, false, null);
}
/**
*
* Creates a new BufferedImage that internally uses the same data as the provided
* GrayU8. The returned BufferedImage will be of type TYPE_BYTE_GRAY.
*
*
*
* NOTE: This only works on images which are not subimages!
*
*
* @param img Input image who's data will be wrapped by the returned BufferedImage.
* @return BufferedImage which shared data with the input image.
*/
public static BufferedImage extractBuffered(GrayU8 img) {
if (img.isSubimage())
throw new IllegalArgumentException("Sub-images are not supported for this operation");
final int width = img.width;
final int height = img.height;
// wrap the byte array
DataBuffer bufferByte = new DataBufferByte(img.data, width * height, 0);
ColorModel colorModel;
int[] bOffs = new int[]{0};
ColorSpace cs = ColorSpace.getInstance(ColorSpace.CS_GRAY);
int[] nBits = {8};
colorModel = new ComponentColorModel(cs, nBits, false, true,
Transparency.OPAQUE,
DataBuffer.TYPE_BYTE);
// Create a raster using the sample model and data buffer
WritableRaster raster = Raster.createInterleavedRaster(
bufferByte, width, height, img.stride, 1, bOffs, new Point(0, 0));
// Combine the color model and raster into a buffered image
return new BufferedImage(colorModel, raster, false, null);
}
/**
* Converts a buffered image into an image of the specified type.
*
* @param src Input BufferedImage which is to be converted
* @param dst The image which it is being converted into
* @param orderRgb If applicable, should it adjust the ordering of each color band to maintain color consistency
*/
public static > void convertFrom(BufferedImage src, T dst , boolean orderRgb) {
if( dst instanceof ImageGray) {
ImageGray sb = (ImageGray)dst;
convertFromSingle(src, sb, (Class) sb.getClass());
} else if( dst instanceof Planar) {
Planar ms = (Planar)dst;
convertFromPlanar(src,ms,orderRgb,ms.getBandType());
} else if( dst instanceof ImageInterleaved ) {
convertFromInterleaved(src, (ImageInterleaved) dst, orderRgb);
} else {
throw new IllegalArgumentException("Unknown type " + dst.getClass().getSimpleName());
}
}
/**
* Converts a buffered image into an image of the specified type.
*
* @param src Input BufferedImage which is to be converted
* @param orderRgb If applicable, should it adjust the ordering of each color band to maintain color consistency
* @param imageType Type of image it is to be converted into
* @return The image
*/
public static > T convertFrom(BufferedImage src , boolean orderRgb , ImageType imageType) {
T out = imageType.createImage(src.getWidth(),src.getHeight());
switch( imageType.getFamily() ) {
case GRAY:
convertFromSingle(src, (ImageGray)out, imageType.getImageClass());
break;
case PLANAR:
convertFromPlanar(src, (Planar) out, orderRgb, imageType.getImageClass());
break;
case INTERLEAVED:
convertFromInterleaved(src, (ImageInterleaved) out, orderRgb);
break;
default:
throw new RuntimeException("Not supported yet");
}
return out;
}
public static > T convertFrom(BufferedImage src , boolean orderRgb , T output ) {
ImageType imageType = output.getImageType();
switch( imageType.getFamily() ) {
case GRAY:
convertFromSingle(src, (ImageGray)output, imageType.getImageClass());
break;
case PLANAR:
convertFromPlanar(src, (Planar) output, orderRgb, imageType.getImageClass());
break;
case INTERLEAVED:
convertFromInterleaved(src, (ImageInterleaved) output, orderRgb);
break;
default:
throw new RuntimeException("Not supported yet");
}
return output;
}
/**
* Converts a buffered image into an image of the specified type. In a 'dst' image is provided
* it will be used for output, otherwise a new image will be created.
*/
public static > T convertFromSingle(BufferedImage src, T dst, Class type) {
if (type == GrayU8.class) {
return (T) convertFrom(src, (GrayU8) dst);
} else if( GrayI16.class.isAssignableFrom(type) ) {
return (T) convertFrom(src, (GrayI16) dst,(Class)type);
} else if (type == GrayF32.class) {
return (T) convertFrom(src, (GrayF32) dst);
} else {
throw new IllegalArgumentException("Unknown type " + type);
}
}
/**
* Converts the buffered image into an {@link GrayU8}. If the buffered image
* has multiple channels the intensities of each channel are averaged together.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new unsigned image is created.
* @return Converted image.
*/
public static GrayU8 convertFrom(BufferedImage src, GrayU8 dst) {
if (dst != null) {
dst.reshape(src.getWidth(), src.getHeight());
} else {
dst = new GrayU8(src.getWidth(), src.getHeight());
}
try {
DataBuffer buff = src.getRaster().getDataBuffer();
if (buff.getDataType() == DataBuffer.TYPE_BYTE ) {
if( isKnownByteFormat(src) ) {
ConvertRaster.bufferedToGray((DataBufferByte)buff,src.getRaster(), dst);
} else {
ConvertRaster.bufferedToGray(src, dst);
}
} else if (buff.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.bufferedToGray((DataBufferInt)buff,src.getRaster(), dst);
} else {
ConvertRaster.bufferedToGray(src, dst);
}
} catch( java.security.AccessControlException e) {
// Applets don't allow access to the raster()
ConvertRaster.bufferedToGray(src, dst);
}
return dst;
}
/**
* Converts the buffered image into an {@link GrayI16}. If the buffered image
* has multiple channels the intensities of each channel are averaged together.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new unsigned image is created.
* @return Converted image.
*/
public static >T convertFrom(BufferedImage src, T dst , Class type ) {
if (dst != null) {
dst.reshape(src.getWidth(), src.getHeight());
} else {
dst = GeneralizedImageOps.createSingleBand(type, src.getWidth(), src.getHeight());
}
DataBuffer buffer = src.getRaster().getDataBuffer();
if (buffer.getDataType() == DataBuffer.TYPE_USHORT ) {
ConvertRaster.bufferedToGray((DataBufferUShort)buffer, src.getRaster(), dst);
return dst;
}
ConvertRaster.bufferedToGray(src, dst);
return dst;
}
/**
* Converts the buffered image into an {@link GrayF32}. If the buffered image
* has multiple channels the intensities of each channel are averaged together.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new unsigned image is created.
* @return Converted image.
*/
public static GrayF32 convertFrom(BufferedImage src, GrayF32 dst) {
if (dst != null) {
dst.reshape(src.getWidth(), src.getHeight());
} else {
dst = new GrayF32( src.getWidth(), src.getHeight());
}
try {
DataBuffer buff = src.getRaster().getDataBuffer();
if ( buff.getDataType() == DataBuffer.TYPE_BYTE ) {
if( isKnownByteFormat(src) ) {
ConvertRaster.bufferedToGray((DataBufferByte)buff,src.getRaster(), dst);
} else {
ConvertRaster.bufferedToGray(src, dst);
}
} else if (buff.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.bufferedToGray((DataBufferInt)buff, src.getRaster(), dst);
} else {
ConvertRaster.bufferedToGray(src, dst);
}
} catch( java.security.AccessControlException e) {
// Applets don't allow access to the raster()
ConvertRaster.bufferedToGray(src, dst);
}
return dst;
}
/**
* Converts the buffered image into an {@link Planar} image of the specified ype.
*
* @param src Input image. Not modified.
* @param dst Output. The converted image is written to. If null a new unsigned image is created.
* @param orderRgb If applicable, should it adjust the ordering of each color band to maintain color consistency.
* Most of the time you want this to be true.
* @param type Which type of data structure is each band. (GrayU8 or GrayF32)
* @return Converted image.
*/
public static > Planar
convertFromPlanar(BufferedImage src, Planar dst , boolean orderRgb , Class type )
{
if( src == null )
throw new IllegalArgumentException("src is null!");
if (dst != null) {
dst.reshape(src.getWidth(), src.getHeight());
}
try {
WritableRaster raster = src.getRaster();
int numBands;
if( !isKnownByteFormat(src) )
numBands = 3;
else
numBands = raster.getNumBands();
if( dst == null)
dst = new Planar<>(type, src.getWidth(), src.getHeight(), numBands);
else if( dst.getNumBands() != numBands )
dst.setNumberOfBands(numBands);
DataBuffer srcBuff = src.getRaster().getDataBuffer();
if( type == GrayU8.class ) {
if (srcBuff.getDataType() == DataBuffer.TYPE_BYTE &&
isKnownByteFormat(src) ) {
if( src.getType() == BufferedImage.TYPE_BYTE_GRAY) {
for( int i = 0; i < dst.getNumBands(); i++ )
ConvertRaster.bufferedToGray(src, ((Planar) dst).getBand(i));
} else {
ConvertRaster.bufferedToMulti_U8((DataBufferByte)srcBuff, src.getRaster(), (Planar) dst);
}
} else if (srcBuff.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.bufferedToMulti_U8((DataBufferInt)srcBuff, src.getRaster(), (Planar) dst);
} else {
ConvertRaster.bufferedToMulti_U8(src, (Planar) dst);
}
} else if( type == GrayF32.class ) {
if (srcBuff.getDataType() == DataBuffer.TYPE_BYTE &&
isKnownByteFormat(src) ) {
if( src.getType() == BufferedImage.TYPE_BYTE_GRAY) {
for( int i = 0; i < dst.getNumBands(); i++ )
ConvertRaster.bufferedToGray(src,((Planar)dst).getBand(i));
} else {
ConvertRaster.bufferedToMulti_F32( (DataBufferByte)srcBuff, src.getRaster(), (Planar) dst);
}
} else if (srcBuff.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.bufferedToMulti_F32((DataBufferInt)srcBuff, src.getRaster(), (Planar) dst);
} else {
ConvertRaster.bufferedToMulti_F32(src, (Planar) dst);
}
} else {
throw new IllegalArgumentException("Band type not supported yet");
}
} catch( java.security.AccessControlException e) {
// Applets don't allow access to the raster()
if( dst == null )
dst = new Planar<>(type, src.getWidth(), src.getHeight(), 3);
else {
dst.setNumberOfBands(3);
}
if( type == GrayU8.class ) {
ConvertRaster.bufferedToMulti_U8(src, (Planar) dst);
} else if( type == GrayF32.class ) {
ConvertRaster.bufferedToMulti_F32(src, (Planar)dst);
}
}
// if requested, ensure the ordering of the bands
if( orderRgb ) {
orderBandsIntoRGB(dst,src);
}
return dst;
}
public static T convertFrom(BufferedImage src, Class type, boolean orderRgb) {
T dst;
if( ImageGray.class.isAssignableFrom(type)) {
dst = (T)convertFromSingle(src, null, type);
} else if ( ImageInterleaved.class.isAssignableFrom(type) ){
dst = (T) GeneralizedImageOps.createInterleaved(type, 1, 1, 3);
convertFromInterleaved(src,(ImageInterleaved)dst,orderRgb);
} else {
dst = (T)new Planar<>(GrayU8.class,1,1,3);
convertFrom(src,dst, orderRgb);
}
return dst;
}
public static void convertFromInterleaved(BufferedImage src, ImageInterleaved dst, boolean orderRgb)
{
if (src == null)
throw new IllegalArgumentException("src is null!");
try {
WritableRaster raster = src.getRaster();
int numBands;
if( !isKnownByteFormat(src) )
numBands = 3;
else
numBands = raster.getNumBands();
dst.setNumberOfBands(numBands);
dst.reshape(src.getWidth(), src.getHeight());
DataBuffer buffer = src.getRaster().getDataBuffer();
if( dst instanceof InterleavedU8 ) {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE ){
if(isKnownByteFormat(src)) {
if (src.getType() == BufferedImage.TYPE_BYTE_GRAY) {
ConvertRaster.bufferedToGray(src, (InterleavedU8) dst);
} else {
ConvertRaster.bufferedToInterleaved((DataBufferByte)buffer,src.getRaster(), (InterleavedU8) dst);
}
} else {
ConvertRaster.bufferedToInterleaved(src, (InterleavedU8) dst);
}
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.bufferedToInterleaved((DataBufferInt)buffer, src.getRaster(), (InterleavedU8) dst);
} else {
ConvertRaster.bufferedToInterleaved(src, (InterleavedU8) dst);
}
} else if( dst instanceof InterleavedF32 ) {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE ) {
if(isKnownByteFormat(src)) {
if (src.getType() == BufferedImage.TYPE_BYTE_GRAY) {
ConvertRaster.bufferedToGray(src, (InterleavedF32) dst);
} else {
ConvertRaster.bufferedToInterleaved((DataBufferByte)buffer,src.getRaster(), (InterleavedF32) dst);
}
} else {
ConvertRaster.bufferedToInterleaved(src, (InterleavedF32) dst);
}
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.bufferedToInterleaved((DataBufferInt)buffer, src.getRaster(), (InterleavedF32) dst);
} else {
ConvertRaster.bufferedToInterleaved(src, (InterleavedF32) dst);
}
} else {
throw new IllegalArgumentException("Data type not supported yet");
}
} catch( java.security.AccessControlException e) {
// force the number of bands to be something valid
if( dst.getNumBands() != 3 || dst.getNumBands() != 1 ) {
dst.setNumberOfBands(3);
}
dst.reshape(src.getWidth(), src.getHeight());
// Applets don't allow access to the raster()
if( dst instanceof InterleavedU8 ) {
ConvertRaster.bufferedToInterleaved(src, (InterleavedU8) dst);
} else if( dst instanceof InterleavedF32 ) {
ConvertRaster.bufferedToInterleaved(src, (InterleavedF32) dst);
} else {
throw new IllegalArgumentException("Unsupported dst image type");
}
}
// if requested, ensure the ordering of the bands
if( orderRgb ) {
orderBandsIntoRGB(dst,src);
}
}
/**
*
* Converts an image into a BufferedImage. The best way to think of this function is that it's a mindless
* typecast. If you don't provide an output image then it will create one. However there isn't always a direct
* equivalent between a BoofCV image and BufferedImage internal type. A "reasonable" choice will be made, but
* for your application it might not be a good choice.
*
*
* @param src Input image. Pixels must have a value from 0 to 255.
* @param dst Where the converted image is written to. If null a new image is created. See comment above about type.
* @param orderRgb If applicable, should it change the order of the color bands (assumed RGB or ARGB) into the
* order based on BufferedImage.TYPE. Most of the time you want this to be true.
* @return Converted image.
*/
public static BufferedImage convertTo( ImageBase src, BufferedImage dst, boolean orderRgb ) {
if( src instanceof ImageGray) {
if( GrayU8.class == src.getClass() ) {
return convertTo((GrayU8)src,dst);
} else if( GrayI16.class.isInstance(src) ) {
return convertTo((GrayI16)src,dst);
} else if( GrayF32.class == src.getClass() ) {
return convertTo((GrayF32)src,dst);
} else {
throw new IllegalArgumentException("ImageGray type is not yet supported: "+src.getClass().getSimpleName());
}
} else if( src instanceof Planar) {
Planar ms = (Planar)src;
if( GrayU8.class == ms.getBandType() ) {
return convertTo_U8((Planar) ms, dst, orderRgb);
} else if( GrayF32.class == ms.getBandType() ) {
return convertTo_F32((Planar) ms, dst, orderRgb);
} else {
throw new IllegalArgumentException("Planar type is not yet supported: "+ ms.getBandType().getSimpleName());
}
} else if( src instanceof ImageInterleaved ) {
if( InterleavedU8.class == src.getClass() ) {
return convertTo((InterleavedU8)src,dst,orderRgb);
} else if( InterleavedF32.class == src.getClass() ) {
return convertTo((InterleavedF32)src,dst,orderRgb);
} else {
throw new IllegalArgumentException("ImageGray type is not yet supported: "+src.getClass().getSimpleName());
}
}
throw new IllegalArgumentException("Image type is not yet supported: "+src.getClass().getSimpleName());
}
/**
* Converts a {@link GrayU8} into a BufferedImage. If the buffered image
* has multiple channels then the input image is copied into each channel.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new image is created.
* @return Converted image.
*/
public static BufferedImage convertTo(GrayU8 src, BufferedImage dst) {
dst = checkInputs(src, dst);
DataBuffer buffer = dst.getRaster().getDataBuffer();
try {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(dst) ) {
ConvertRaster.grayToBuffered(src, (DataBufferByte)buffer, dst.getRaster());
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.grayToBuffered(src, (DataBufferInt)buffer, dst.getRaster());
} else {
ConvertRaster.grayToBuffered(src, dst);
}
// hack so that it knows the buffer has been modified
dst.setRGB(0,0,dst.getRGB(0,0));
} catch( java.security.AccessControlException e) {
ConvertRaster.grayToBuffered(src, dst);
}
return dst;
}
/**
* Converts a {@link GrayI16} into a BufferedImage. If the buffered image
* has multiple channels then the input image is copied into each channel.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new image is created.
* @return Converted image.
*/
public static BufferedImage convertTo(GrayI16 src, BufferedImage dst) {
dst = checkInputs(src, dst);
DataBuffer buffer = dst.getRaster().getDataBuffer();
try {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(dst) ) {
ConvertRaster.grayToBuffered(src, (DataBufferByte)buffer, dst.getRaster());
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.grayToBuffered(src, (DataBufferInt)buffer, dst.getRaster());
} else if( buffer.getDataType() == DataBuffer.TYPE_USHORT ) {
ConvertRaster.grayToBuffered(src, (DataBufferUShort)buffer, dst.getRaster());
} else {
ConvertRaster.grayToBuffered(src, dst);
}
// hack so that it knows the buffer has been modified
dst.setRGB(0,0,dst.getRGB(0,0));
} catch( java.security.AccessControlException e) {
ConvertRaster.grayToBuffered(src, dst);
}
return dst;
}
/**
* Converts the buffered image into an {@link GrayF32}. If the buffered image
* has multiple channels then the input image is copied into each channel. The floating
* point image is assumed to be between 0 and 255.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new image is created.
* @return Converted image.
*/
public static BufferedImage convertTo(GrayF32 src, BufferedImage dst) {
dst = checkInputs(src, dst);
DataBuffer buffer = dst.getRaster().getDataBuffer();
try {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(dst) ) {
ConvertRaster.grayToBuffered(src, (DataBufferByte)buffer, dst.getRaster());
} else if ( buffer.getDataType() == DataBuffer.TYPE_INT ) {
ConvertRaster.grayToBuffered(src, (DataBufferInt)buffer, dst.getRaster());
} else {
ConvertRaster.grayToBuffered(src, dst);
}
// hack so that it knows the buffer has been modified
dst.setRGB(0,0,dst.getRGB(0,0));
} catch( java.security.AccessControlException e) {
ConvertRaster.grayToBuffered(src, dst);
}
return dst;
}
/**
* Converts a {@link Planar} {@link GrayU8} into a BufferedImage.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new image is created.
* @param orderRgb If applicable, should it change the order of the color bands (assumed RGB or ARGB) into the
* order based on BufferedImage.TYPE. Most of the time you want this to be true.
* @return Converted image.
*/
public static BufferedImage convertTo_U8(Planar src, BufferedImage dst, boolean orderRgb ) {
dst = checkInputs(src, dst);
if( orderRgb ) {
src = orderBandsIntoBuffered(src, dst);
}
DataBuffer buffer = dst.getRaster().getDataBuffer();
try {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(dst) ) {
ConvertRaster.multToBuffered_U8(src, (DataBufferByte)buffer, dst.getRaster());
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.multToBuffered_U8(src, (DataBufferInt)buffer, dst.getRaster());
} else {
ConvertRaster.multToBuffered_U8(src, dst);
}
// hack so that it knows the buffer has been modified
dst.setRGB(0,0,dst.getRGB(0,0));
} catch( java.security.AccessControlException e) {
ConvertRaster.multToBuffered_U8(src, dst);
}
return dst;
}
/**
* Converts a {@link Planar} {@link GrayF32} into a BufferedImage.
*
* @param src Input image.
* @param dst Where the converted image is written to. If null a new image is created.
* @param orderRgb If applicable, should it change the order of the color bands (assumed RGB or ARGB) into the
* order based on BufferedImage.TYPE. Most of the time you want this to be true.
* @return Converted image.
*/
public static BufferedImage convertTo_F32(Planar src, BufferedImage dst, boolean orderRgb) {
dst = checkInputs(src, dst);
if( orderRgb ) {
src = orderBandsIntoBuffered(src, dst);
}
DataBuffer buffer = dst.getRaster().getDataBuffer();
try {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(dst) ) {
ConvertRaster.multToBuffered_F32(src, (DataBufferByte)buffer, dst.getRaster());
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.multToBuffered_F32(src, (DataBufferInt)buffer, dst.getRaster());
} else {
ConvertRaster.multToBuffered_F32(src, dst);
}
// hack so that it knows the buffer has been modified
dst.setRGB(0,0,dst.getRGB(0,0));
} catch( java.security.AccessControlException e) {
ConvertRaster.multToBuffered_F32(src, dst);
}
return dst;
}
public static BufferedImage convertTo( InterleavedU8 src, BufferedImage dst, boolean orderRgb ) {
dst = checkInputs(src, dst);
DataBuffer buffer = dst.getRaster().getDataBuffer();
try {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(dst) ) {
ConvertRaster.interleavedToBuffered(src, (DataBufferByte)buffer, dst.getRaster());
if( orderRgb )
orderBandsBufferedFromRGB((DataBufferByte)buffer, dst.getRaster(),dst.getType());
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.interleavedToBuffered(src, (DataBufferInt)buffer, dst.getRaster());
if( orderRgb )
orderBandsBufferedFromRGB((DataBufferInt)buffer, dst.getRaster(), dst.getType());
} else {
ConvertRaster.interleavedToBuffered(src, dst);
}
// hack so that it knows the buffer has been modified
dst.setRGB(0,0,dst.getRGB(0,0));
} catch( java.security.AccessControlException e) {
ConvertRaster.interleavedToBuffered(src, dst);
}
return dst;
}
public static BufferedImage convertTo( InterleavedF32 src, BufferedImage dst, boolean orderRgb ) {
dst = checkInputs(src, dst);
DataBuffer buffer = dst.getRaster().getDataBuffer();
try {
if (buffer.getDataType() == DataBuffer.TYPE_BYTE && isKnownByteFormat(dst) ) {
ConvertRaster.interleavedToBuffered(src, (DataBufferByte)buffer, dst.getRaster());
if( orderRgb )
orderBandsBufferedFromRGB((DataBufferByte)buffer, dst.getRaster(),dst.getType());
} else if (buffer.getDataType() == DataBuffer.TYPE_INT) {
ConvertRaster.interleavedToBuffered(src, (DataBufferInt)buffer, dst.getRaster());
if( orderRgb )
orderBandsBufferedFromRGB((DataBufferInt)buffer, dst.getRaster(), dst.getType());
} else {
ConvertRaster.interleavedToBuffered(src, dst);
}
// hack so that it knows the buffer has been modified
dst.setRGB(0,0,dst.getRGB(0,0));
} catch( java.security.AccessControlException e) {
ConvertRaster.interleavedToBuffered(src, dst);
}
return dst;
}
/**
* If null the dst is declared, otherwise it checks to see if the 'dst' as the same shape as 'src'.
*/
public static BufferedImage checkInputs(ImageBase src, BufferedImage dst) {
if (dst != null) {
if (dst.getWidth() != src.getWidth() || dst.getHeight() != src.getHeight()) {
throw new IllegalArgumentException("Shapes do not match: "+
"src = ( "+src.width+" , "+src.height+" ) " +
"dst = ( "+dst.getWidth()+" , "+dst.getHeight()+" )");
}
} else {
if( GrayI8.class.isInstance(src))
dst = new BufferedImage(src.getWidth(), src.getHeight(), BufferedImage.TYPE_BYTE_GRAY);
else if( GrayF.class.isInstance(src) )
// no good equivalent. Just assume the image is a regular gray scale image
// with pixel values from 0 to 255
dst = new BufferedImage(src.getWidth(), src.getHeight(), BufferedImage.TYPE_BYTE_GRAY);
// throw new RuntimeException("Fail!");
else if( GrayI.class.isInstance(src))
// no good equivalent. I'm giving it the biggest pixel for the range
dst = new BufferedImage(src.getWidth(), src.getHeight(), BufferedImage.TYPE_USHORT_GRAY);
else
dst = new BufferedImage(src.getWidth(), src.getHeight(), BufferedImage.TYPE_INT_RGB);
}
return dst;
}
/**
* Draws the component into a BufferedImage.
*
* @param comp The component being drawn into an image.
* @param storage if not null the component is drawn into it, if null a new BufferedImage is created.
* @return image of the component
*/
public static BufferedImage convertTo(JComponent comp, BufferedImage storage) {
if (storage == null)
storage = new BufferedImage(comp.getWidth(), comp.getHeight(), BufferedImage.TYPE_INT_RGB);
Graphics2D g2 = storage.createGraphics();
comp.paintComponents(g2);
return storage;
}
/**
* Returns a new image with the color bands in the appropriate ordering. The returned image will
* reference the original image's image arrays.
*/
public static Planar orderBandsIntoBuffered(Planar src, BufferedImage dst) {
// see if no change is required
if( dst.getType() == BufferedImage.TYPE_INT_RGB )
return src;
Planar tmp = new Planar(src.type, src.getNumBands());
tmp.width = src.width;
tmp.height = src.height;
tmp.stride = src.stride;
tmp.startIndex = src.startIndex;
for( int i = 0; i < src.getNumBands(); i++ ) {
tmp.bands[i] = src.bands[i];
}
orderBandsBufferedFromRgb(tmp, dst);
return tmp;
}
/**
* The image the BufferedImage was created from had RGB or RGBA color order. This swaps the bytes around
* to put it into the expected local format
*/
public static void orderBandsBufferedFromRGB( DataBufferByte buffer , WritableRaster raster , int type ) {
int height = raster.getHeight();
int width = raster.getWidth();
int stride = ConvertRaster.stride(raster);
int offset = ConvertRaster.getOffset(raster);
byte data[] = buffer.getData();
if( BufferedImage.TYPE_3BYTE_BGR == type ) {
for (int y = 0; y < height; y++) {
int index = offset + y*stride;
for (int x = 0; x < width; x++) {
byte tmp = data[index];
data[index] = data[index+2];
data[index+2] = tmp;
index += 3;
}
}
} else if( BufferedImage.TYPE_4BYTE_ABGR == type ) {
for (int y = 0; y < height; y++) {
int index = offset + y*stride;
for (int x = 0; x < width; x++) {
byte tmp0 = data[index];
byte tmp1 = data[index+1];
data[index] = data[index+3];
data[index+1] = data[index+2];
data[index+2] = tmp1;
data[index+3] = tmp0;
index += 4;
}
}
} else {
throw new IllegalArgumentException("Unsupported buffered image type");
}
}
/**
* The image the BufferedImage was created from had RGB or RGBA color order. This swaps the bytes around
* to put it into the expected local format
*/
public static void orderBandsBufferedFromRGB( DataBufferInt buffer, WritableRaster raster , int type ) {
if( BufferedImage.TYPE_INT_RGB == type )
return;
int height = raster.getHeight();
int width = raster.getWidth();
int stride = ConvertRaster.stride(raster);
int offset = ConvertRaster.getOffset(raster);
int data[] = buffer.getData();
if( BufferedImage.TYPE_INT_BGR == type ) {
for (int y = 0; y < height; y++) {
int index = offset + y*stride;
for (int x = 0; x < width; x++, index++) {
int tmp = data[index];
int c1 = tmp & 0xFF;
int c2 = (tmp >> 8) & 0xFF;
int c3 = (tmp >> 16) & 0xFF;
data[index] = c1 << 16 | c2 << 8 | c3;
}
}
} else if( BufferedImage.TYPE_INT_ARGB == type ) {
for (int y = 0; y < height; y++) {
int index = offset + y*stride;
for (int x = 0; x < width; x++, index++) {
int tmp = data[index];
int c1 = tmp & 0xFF;
int c2 = (tmp >> 8) & 0xFF;
int c3 = (tmp >> 16) & 0xFF;
int c4 = (tmp >> 24) & 0xFF;
data[index] = c1 << 24 | c4 << 16 | c3 << 8 | c2;
}
}
} else {
throw new IllegalArgumentException("Unsupported buffered image type");
}
}
/**
* If a Planar was created from a BufferedImage its colors might not be in the expected order.
* Invoking this function ensures that the image will have the expected ordering. For images with
* 3 bands it will be RGB and for 4 bands it will be ARGB.
*/
public static >
void orderBandsIntoRGB(Planar image , BufferedImage input ) {
boolean swap = swapBandOrder(input);
// Output formats are: RGB and RGBA
if( swap ) {
if( image.getNumBands() == 3 ) {
int bufferedImageType = input.getType();
if( bufferedImageType == BufferedImage.TYPE_3BYTE_BGR ||
bufferedImageType == BufferedImage.TYPE_INT_BGR )
{
T tmp = image.getBand(0);
image.bands[0] = image.getBand(2);
image.bands[2] = tmp;
}
} else if( image.getNumBands() == 4 ) {
T[] temp = (T[])Array.newInstance(image.getBandType(),4);
int bufferedImageType = input.getType();
if( bufferedImageType == BufferedImage.TYPE_INT_ARGB ) {
temp[0] = image.getBand(1);
temp[1] = image.getBand(2);
temp[2] = image.getBand(3);
temp[3] = image.getBand(0);
} else if( bufferedImageType == BufferedImage.TYPE_4BYTE_ABGR ) {
temp[0] = image.getBand(3);
temp[1] = image.getBand(2);
temp[2] = image.getBand(1);
temp[3] = image.getBand(0);
}
image.bands[0] = temp[0];
image.bands[1] = temp[1];
image.bands[2] = temp[2];
image.bands[3] = temp[3];
}
}
}
public static >
void orderBandsBufferedFromRgb(Planar image, BufferedImage input) {
boolean swap = swapBandOrder(input);
// Output formats are: RGB and RGBA
if( swap ) {
if( image.getNumBands() == 3 ) {
int bufferedImageType = input.getType();
if( bufferedImageType == BufferedImage.TYPE_3BYTE_BGR ||
bufferedImageType == BufferedImage.TYPE_INT_BGR ) {
T tmp = image.getBand(0);
image.bands[0] = image.getBand(2);
image.bands[2] = tmp;
}
} else if( image.getNumBands() == 4 ) {
T[] temp = (T[])Array.newInstance(image.getBandType(),4);
int bufferedImageType = input.getType();
if( bufferedImageType == BufferedImage.TYPE_INT_ARGB ) {
temp[0] = image.getBand(3);
temp[1] = image.getBand(0);
temp[2] = image.getBand(1);
temp[3] = image.getBand(2);
} else if( bufferedImageType == BufferedImage.TYPE_4BYTE_ABGR ) {
temp[0] = image.getBand(3);
temp[1] = image.getBand(2);
temp[2] = image.getBand(1);
temp[3] = image.getBand(0);
}
image.bands[0] = temp[0];
image.bands[1] = temp[1];
image.bands[2] = temp[2];
image.bands[3] = temp[3];
}
}
}
public static void orderBandsIntoRGB( ImageInterleaved image , BufferedImage input ) {
if( image instanceof InterleavedU8 ) {
orderBandsIntoRGB((InterleavedU8)image,input);
} else if( image instanceof InterleavedF32 ) {
orderBandsIntoRGB((InterleavedF32)image,input);
} else {
throw new IllegalArgumentException("Unsupported interleaved type");
}
}
public static void orderBandsIntoRGB( InterleavedU8 image , BufferedImage input ) {
boolean swap = swapBandOrder(input);
// Output formats are: RGB and RGBA
if( swap ) {
if( image.getNumBands() == 3 ) {
for (int y = 0; y < image.height; y++) {
int index = image.startIndex + y*image.stride;
int indexEnd = index + image.width*3;
while( index < indexEnd ) {
byte tmp = image.data[index+2];
image.data[index+2] = image.data[index];
image.data[index] = tmp;
index += 3;
}
}
} else if( image.getNumBands() == 4 ) {
int bufferedImageType = input.getType();
if( bufferedImageType == BufferedImage.TYPE_INT_ARGB ) {
for (int y = 0; y < image.height; y++) {
int index = image.startIndex + y*image.stride;
int indexEnd = index + image.width*3;
while( index < indexEnd ) {
byte tmp = image.data[index];
image.data[index] = image.data[index+1];
image.data[index+1] = image.data[index+2];
image.data[index+2] = image.data[index+3];
image.data[index+3] = tmp;
index += 4;
}
}
} else if( bufferedImageType == BufferedImage.TYPE_4BYTE_ABGR ) {
for (int y = 0; y < image.height; y++) {
int index = image.startIndex + y*image.stride;
int indexEnd = index + image.width*3;
while( index < indexEnd ) {
byte tmp1 = image.data[index+1];
byte tmp0 = image.data[index];
image.data[index] = image.data[index+3];
image.data[index+1] = image.data[index+2];
image.data[index+2] = tmp1;
image.data[index+3] = tmp0;
index += 4;
}
}
}
}
}
}
public static void orderBandsIntoRGB( InterleavedF32 image , BufferedImage input ) {
boolean swap = swapBandOrder(input);
// Output formats are: RGB and RGBA
if( swap ) {
if( image.getNumBands() == 3 ) {
for (int y = 0; y < image.height; y++) {
int index = image.startIndex + y*image.stride;
int indexEnd = index + image.width*3;
while( index < indexEnd ) {
float tmp = image.data[index+2];
image.data[index+2] = image.data[index];
image.data[index] = tmp;
index += 3;
}
}
} else if( image.getNumBands() == 4 ) {
int bufferedImageType = input.getType();
if( bufferedImageType == BufferedImage.TYPE_INT_ARGB ) {
for (int y = 0; y < image.height; y++) {
int index = image.startIndex + y*image.stride;
int indexEnd = index + image.width*3;
while( index < indexEnd ) {
float tmp = image.data[index];
image.data[index] = image.data[index+1];
image.data[index+1] = image.data[index+2];
image.data[index+2] = image.data[index+3];
image.data[index+3] = tmp;
index += 4;
}
}
} else if( bufferedImageType == BufferedImage.TYPE_4BYTE_ABGR ) {
for (int y = 0; y < image.height; y++) {
int index = image.startIndex + y*image.stride;
int indexEnd = index + image.width*3;
while( index < indexEnd ) {
float tmp1 = image.data[index+1];
float tmp0 = image.data[index];
image.data[index] = image.data[index+3];
image.data[index+1] = image.data[index+2];
image.data[index+2] = tmp1;
image.data[index+3] = tmp0;
index += 4;
}
}
}
}
}
}
private static boolean swapBandOrder(BufferedImage input) {
// see if access to the raster is restricted or not
int bufferedImageType = input.getType();
if( bufferedImageType == BufferedImage.TYPE_3BYTE_BGR ||
bufferedImageType == BufferedImage.TYPE_INT_BGR ||
bufferedImageType == BufferedImage.TYPE_INT_ARGB ||
bufferedImageType == BufferedImage.TYPE_4BYTE_ABGR ) {
return true;
}
return false;
}
/**
* Checks to see if it is a known byte format
*/
public static boolean isKnownByteFormat( BufferedImage image ) {
int type = image.getType();
return type != BufferedImage.TYPE_BYTE_INDEXED &&
type != BufferedImage.TYPE_BYTE_BINARY &&
type != BufferedImage.TYPE_CUSTOM;
}
/**
* Checks to see if the input image is a subImage().
* @param img
* @return
*/
public static boolean isSubImage( BufferedImage img ) {
return img.getRaster().getParent() != null;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy