mirror of
https://github.com/etlegacy/etlegacy-libs.git
synced 2025-02-24 20:21:11 +00:00
557 lines
21 KiB
Java
557 lines
21 KiB
Java
|
/*
|
||
|
* Copyright (C)2011-2014 D. R. Commander. All Rights Reserved.
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
*
|
||
|
* - Redistributions of source code must retain the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer.
|
||
|
* - Redistributions in binary form must reproduce the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer in the documentation
|
||
|
* and/or other materials provided with the distribution.
|
||
|
* - Neither the name of the libjpeg-turbo Project nor the names of its
|
||
|
* contributors may be used to endorse or promote products derived from this
|
||
|
* software without specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS",
|
||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
|
||
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||
|
*/
|
||
|
|
||
|
package org.libjpegturbo.turbojpeg;
|
||
|
|
||
|
import java.awt.image.*;
|
||
|
import java.nio.*;
|
||
|
|
||
|
/**
|
||
|
* TurboJPEG compressor
|
||
|
*/
|
||
|
public class TJCompressor {
|
||
|
|
||
|
private static final String NO_ASSOC_ERROR =
|
||
|
"No source image is associated with this instance";
|
||
|
|
||
|
/**
|
||
|
* Create a TurboJPEG compressor instance.
|
||
|
*/
|
||
|
public TJCompressor() throws Exception {
|
||
|
init();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create a TurboJPEG compressor instance and associate the uncompressed
|
||
|
* source image stored in <code>srcImage</code> with the newly created
|
||
|
* instance.
|
||
|
*
|
||
|
* @param srcImage see {@link #setSourceImage} for description
|
||
|
*
|
||
|
* @param x see {@link #setSourceImage} for description
|
||
|
*
|
||
|
* @param y see {@link #setSourceImage} for description
|
||
|
*
|
||
|
* @param width see {@link #setSourceImage} for description
|
||
|
*
|
||
|
* @param pitch see {@link #setSourceImage} for description
|
||
|
*
|
||
|
* @param height see {@link #setSourceImage} for description
|
||
|
*
|
||
|
* @param pixelFormat pixel format of the source image (one of
|
||
|
* {@link TJ#PF_RGB TJ.PF_*})
|
||
|
*/
|
||
|
public TJCompressor(byte[] srcImage, int x, int y, int width, int pitch,
|
||
|
int height, int pixelFormat) throws Exception {
|
||
|
setSourceImage(srcImage, x, y, width, pitch, height, pixelFormat);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @deprecated Use
|
||
|
* {@link #TJCompressor(byte[], int, int, int, int, int, int)} instead.
|
||
|
*/
|
||
|
@Deprecated
|
||
|
public TJCompressor(byte[] srcImage, int width, int pitch, int height,
|
||
|
int pixelFormat) throws Exception {
|
||
|
setSourceImage(srcImage, width, pitch, height, pixelFormat);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Associate an uncompressed source image with this compressor instance.
|
||
|
*
|
||
|
* @param srcImage image buffer containing RGB or grayscale pixels to be
|
||
|
* compressed or encoded
|
||
|
*
|
||
|
* @param x x offset (in pixels) of the region in the source image from which
|
||
|
* the JPEG or YUV image should be compressed/encoded
|
||
|
*
|
||
|
* @param y y offset (in pixels) of the region in the source image from which
|
||
|
* the JPEG or YUV image should be compressed/encoded
|
||
|
*
|
||
|
* @param width width (in pixels) of the region in the source image from
|
||
|
* which the JPEG or YUV image should be compressed/encoded
|
||
|
*
|
||
|
* @param pitch bytes per line of the source image. Normally, this should be
|
||
|
* <code>width * TJ.pixelSize(pixelFormat)</code> if the source image is
|
||
|
* unpadded, but you can use this parameter to, for instance, specify that
|
||
|
* the scanlines in the source image are padded to a 4-byte boundary or to
|
||
|
* compress/encode a JPEG or YUV image from a region of a larger source
|
||
|
* image. You can also be clever and use this parameter to skip lines, etc.
|
||
|
* Setting this parameter to 0 is the equivalent of setting it to
|
||
|
* <code>width * TJ.pixelSize(pixelFormat)</code>.
|
||
|
*
|
||
|
* @param height height (in pixels) of the region in the source image from
|
||
|
* which the JPEG or YUV image should be compressed/encoded
|
||
|
*
|
||
|
* @param pixelFormat pixel format of the source image (one of
|
||
|
* {@link TJ#PF_RGB TJ.PF_*})
|
||
|
*/
|
||
|
public void setSourceImage(byte[] srcImage, int x, int y, int width,
|
||
|
int pitch, int height, int pixelFormat)
|
||
|
throws Exception {
|
||
|
if (handle == 0) init();
|
||
|
if (srcImage == null || x < 0 || y < 0 || width < 1 || height < 1 ||
|
||
|
pitch < 0 || pixelFormat < 0 || pixelFormat >= TJ.NUMPF)
|
||
|
throw new Exception("Invalid argument in setSourceImage()");
|
||
|
srcBuf = srcImage;
|
||
|
srcWidth = width;
|
||
|
if (pitch == 0)
|
||
|
srcPitch = width * TJ.getPixelSize(pixelFormat);
|
||
|
else
|
||
|
srcPitch = pitch;
|
||
|
srcHeight = height;
|
||
|
srcPixelFormat = pixelFormat;
|
||
|
srcX = x;
|
||
|
srcY = y;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @deprecated Use
|
||
|
* {@link #setSourceImage(byte[], int, int, int, int, int, int)} instead.
|
||
|
*/
|
||
|
@Deprecated
|
||
|
public void setSourceImage(byte[] srcImage, int width, int pitch,
|
||
|
int height, int pixelFormat) throws Exception {
|
||
|
setSourceImage(srcImage, 0, 0, width, pitch, height, pixelFormat);
|
||
|
srcX = srcY = -1;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Set the level of chrominance subsampling for subsequent compress/encode
|
||
|
* operations.
|
||
|
*
|
||
|
* @param newSubsamp the level of chrominance subsampling to use in
|
||
|
* subsequent compress/encode operations (one of
|
||
|
* {@link TJ#SAMP_444 TJ.SAMP_*})
|
||
|
*/
|
||
|
public void setSubsamp(int newSubsamp) throws Exception {
|
||
|
if (newSubsamp < 0 || newSubsamp >= TJ.NUMSAMP)
|
||
|
throw new Exception("Invalid argument in setSubsamp()");
|
||
|
subsamp = newSubsamp;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the JPEG image quality level for subsequent compress operations.
|
||
|
*
|
||
|
* @param quality the new JPEG image quality level (1 to 100, 1 = worst,
|
||
|
* 100 = best)
|
||
|
*/
|
||
|
public void setJPEGQuality(int quality) throws Exception {
|
||
|
if (quality < 1 || quality > 100)
|
||
|
throw new Exception("Invalid argument in setJPEGQuality()");
|
||
|
jpegQuality = quality;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compress the uncompressed source image associated with this compressor
|
||
|
* instance and output a JPEG image to the given destination buffer.
|
||
|
*
|
||
|
* @param dstBuf buffer that will receive the JPEG image. Use
|
||
|
* {@link TJ#bufSize} to determine the maximum size for this buffer based on
|
||
|
* the source image's width and height and the desired level of chrominance
|
||
|
* subsampling.
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*/
|
||
|
public void compress(byte[] dstBuf, int flags) throws Exception {
|
||
|
if (dstBuf == null || flags < 0)
|
||
|
throw new Exception("Invalid argument in compress()");
|
||
|
if (srcBuf == null)
|
||
|
throw new Exception(NO_ASSOC_ERROR);
|
||
|
if (jpegQuality < 0)
|
||
|
throw new Exception("JPEG Quality not set");
|
||
|
if (subsamp < 0)
|
||
|
throw new Exception("Subsampling level not set");
|
||
|
if (srcX >= 0 && srcY >= 0)
|
||
|
compressedSize = compress(srcBuf, srcX, srcY, srcWidth, srcPitch,
|
||
|
srcHeight, srcPixelFormat, dstBuf, subsamp,
|
||
|
jpegQuality, flags);
|
||
|
else
|
||
|
compressedSize = compress(srcBuf, srcWidth, srcPitch, srcHeight,
|
||
|
srcPixelFormat, dstBuf, subsamp, jpegQuality,
|
||
|
flags);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compress the uncompressed source image associated with this compressor
|
||
|
* instance and return a buffer containing a JPEG image.
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*
|
||
|
* @return a buffer containing a JPEG image. The length of this buffer will
|
||
|
* not be equal to the size of the JPEG image. Use {@link
|
||
|
* #getCompressedSize} to obtain the size of the JPEG image.
|
||
|
*/
|
||
|
public byte[] compress(int flags) throws Exception {
|
||
|
if (srcWidth < 1 || srcHeight < 1)
|
||
|
throw new Exception(NO_ASSOC_ERROR);
|
||
|
byte[] buf = new byte[TJ.bufSize(srcWidth, srcHeight, subsamp)];
|
||
|
compress(buf, flags);
|
||
|
return buf;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compress the uncompressed source image stored in <code>srcImage</code>
|
||
|
* and output a JPEG image to the given destination buffer.
|
||
|
*
|
||
|
* @param srcImage a <code>BufferedImage</code> instance containing RGB or
|
||
|
* grayscale pixels to be compressed
|
||
|
*
|
||
|
* @param dstBuf buffer that will receive the JPEG image. Use
|
||
|
* {@link TJ#bufSize} to determine the maximum size for this buffer based on
|
||
|
* the image width, height, and level of chrominance subsampling.
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*/
|
||
|
public void compress(BufferedImage srcImage, byte[] dstBuf, int flags)
|
||
|
throws Exception {
|
||
|
if (srcImage == null || dstBuf == null || flags < 0)
|
||
|
throw new Exception("Invalid argument in compress()");
|
||
|
int width = srcImage.getWidth();
|
||
|
int height = srcImage.getHeight();
|
||
|
int pixelFormat;
|
||
|
boolean intPixels = false;
|
||
|
if (byteOrder == null)
|
||
|
byteOrder = ByteOrder.nativeOrder();
|
||
|
switch(srcImage.getType()) {
|
||
|
case BufferedImage.TYPE_3BYTE_BGR:
|
||
|
pixelFormat = TJ.PF_BGR; break;
|
||
|
case BufferedImage.TYPE_4BYTE_ABGR:
|
||
|
case BufferedImage.TYPE_4BYTE_ABGR_PRE:
|
||
|
pixelFormat = TJ.PF_XBGR; break;
|
||
|
case BufferedImage.TYPE_BYTE_GRAY:
|
||
|
pixelFormat = TJ.PF_GRAY; break;
|
||
|
case BufferedImage.TYPE_INT_BGR:
|
||
|
if (byteOrder == ByteOrder.BIG_ENDIAN)
|
||
|
pixelFormat = TJ.PF_XBGR;
|
||
|
else
|
||
|
pixelFormat = TJ.PF_RGBX;
|
||
|
intPixels = true; break;
|
||
|
case BufferedImage.TYPE_INT_RGB:
|
||
|
case BufferedImage.TYPE_INT_ARGB:
|
||
|
case BufferedImage.TYPE_INT_ARGB_PRE:
|
||
|
if (byteOrder == ByteOrder.BIG_ENDIAN)
|
||
|
pixelFormat = TJ.PF_XRGB;
|
||
|
else
|
||
|
pixelFormat = TJ.PF_BGRX;
|
||
|
intPixels = true; break;
|
||
|
default:
|
||
|
throw new Exception("Unsupported BufferedImage format");
|
||
|
}
|
||
|
WritableRaster wr = srcImage.getRaster();
|
||
|
if (jpegQuality < 0)
|
||
|
throw new Exception("JPEG Quality not set");
|
||
|
if (subsamp < 0)
|
||
|
throw new Exception("Subsampling level not set");
|
||
|
if (intPixels) {
|
||
|
SinglePixelPackedSampleModel sm =
|
||
|
(SinglePixelPackedSampleModel)srcImage.getSampleModel();
|
||
|
int stride = sm.getScanlineStride();
|
||
|
DataBufferInt db = (DataBufferInt)wr.getDataBuffer();
|
||
|
int[] buf = db.getData();
|
||
|
if (srcX >= 0 && srcY >= 0)
|
||
|
compressedSize = compress(buf, srcX, srcY, width, stride, height,
|
||
|
pixelFormat, dstBuf, subsamp, jpegQuality,
|
||
|
flags);
|
||
|
else
|
||
|
compressedSize = compress(buf, width, stride, height, pixelFormat,
|
||
|
dstBuf, subsamp, jpegQuality, flags);
|
||
|
} else {
|
||
|
ComponentSampleModel sm =
|
||
|
(ComponentSampleModel)srcImage.getSampleModel();
|
||
|
int pixelSize = sm.getPixelStride();
|
||
|
if (pixelSize != TJ.getPixelSize(pixelFormat))
|
||
|
throw new Exception("Inconsistency between pixel format and pixel size in BufferedImage");
|
||
|
int pitch = sm.getScanlineStride();
|
||
|
DataBufferByte db = (DataBufferByte)wr.getDataBuffer();
|
||
|
byte[] buf = db.getData();
|
||
|
if (srcX >= 0 && srcY >= 0)
|
||
|
compressedSize = compress(buf, srcX, srcY, width, pitch, height,
|
||
|
pixelFormat, dstBuf, subsamp, jpegQuality,
|
||
|
flags);
|
||
|
else
|
||
|
compressedSize = compress(buf, width, pitch, height, pixelFormat,
|
||
|
dstBuf, subsamp, jpegQuality, flags);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compress the uncompressed source image stored in <code>srcImage</code>
|
||
|
* and return a buffer containing a JPEG image.
|
||
|
*
|
||
|
* @param srcImage a <code>BufferedImage</code> instance containing RGB or
|
||
|
* grayscale pixels to be compressed
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*
|
||
|
* @return a buffer containing a JPEG image. The length of this buffer will
|
||
|
* not be equal to the size of the JPEG image. Use {@link
|
||
|
* #getCompressedSize} to obtain the size of the JPEG image.
|
||
|
*/
|
||
|
public byte[] compress(BufferedImage srcImage, int flags) throws Exception {
|
||
|
int width = srcImage.getWidth();
|
||
|
int height = srcImage.getHeight();
|
||
|
byte[] buf = new byte[TJ.bufSize(width, height, subsamp)];
|
||
|
compress(srcImage, buf, flags);
|
||
|
return buf;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Encode the uncompressed source image associated with this compressor
|
||
|
* instance and output a YUV planar image to the given destination buffer.
|
||
|
* This method uses the accelerated color conversion routines in
|
||
|
* TurboJPEG's underlying codec to produce a planar YUV image that is
|
||
|
* suitable for direct video display. Specifically, if the chrominance
|
||
|
* components are subsampled along the horizontal dimension, then the width
|
||
|
* of the luminance plane is padded to the nearest multiple of 2 in the
|
||
|
* output image (same goes for the height of the luminance plane, if the
|
||
|
* chrominance components are subsampled along the vertical dimension.)
|
||
|
* Also, each line of each plane in the output image is padded to 4 bytes.
|
||
|
* Although this will work with any subsampling option, it is really only
|
||
|
* useful in combination with {@link TJ#SAMP_420}, which produces an image
|
||
|
* compatible with the I420 (AKA "YUV420P") format.
|
||
|
* <p>
|
||
|
* NOTE: Technically, the JPEG format uses the YCbCr colorspace, but per the
|
||
|
* convention of the digital video community, the TurboJPEG API uses "YUV" to
|
||
|
* refer to an image format consisting of Y, Cb, and Cr image planes.
|
||
|
*
|
||
|
* @param dstBuf buffer that will receive the YUV planar image. Use
|
||
|
* {@link TJ#bufSizeYUV} to determine the appropriate size for this buffer
|
||
|
* based on the image width, height, and level of chrominance subsampling.
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*/
|
||
|
public void encodeYUV(byte[] dstBuf, int flags) throws Exception {
|
||
|
if (dstBuf == null || flags < 0)
|
||
|
throw new Exception("Invalid argument in compress()");
|
||
|
if (srcBuf == null)
|
||
|
throw new Exception(NO_ASSOC_ERROR);
|
||
|
if (subsamp < 0)
|
||
|
throw new Exception("Subsampling level not set");
|
||
|
encodeYUV(srcBuf, srcWidth, srcPitch, srcHeight,
|
||
|
srcPixelFormat, dstBuf, subsamp, flags);
|
||
|
compressedSize = TJ.bufSizeYUV(srcWidth, srcHeight, subsamp);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Encode the uncompressed source image associated with this compressor
|
||
|
* instance and return a buffer containing a YUV planar image. See
|
||
|
* {@link #encodeYUV(byte[], int)} for more detail.
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*
|
||
|
* @return a buffer containing a YUV planar image
|
||
|
*/
|
||
|
public byte[] encodeYUV(int flags) throws Exception {
|
||
|
if (srcWidth < 1 || srcHeight < 1)
|
||
|
throw new Exception(NO_ASSOC_ERROR);
|
||
|
if (subsamp < 0)
|
||
|
throw new Exception("Subsampling level not set");
|
||
|
byte[] buf = new byte[TJ.bufSizeYUV(srcWidth, srcHeight, subsamp)];
|
||
|
encodeYUV(buf, flags);
|
||
|
return buf;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Encode the uncompressed source image stored in <code>srcImage</code>
|
||
|
* and output a YUV planar image to the given destination buffer. See
|
||
|
* {@link #encodeYUV(byte[], int)} for more detail.
|
||
|
*
|
||
|
* @param srcImage a <code>BufferedImage</code> instance containing RGB or
|
||
|
* grayscale pixels to be encoded
|
||
|
*
|
||
|
* @param dstBuf buffer that will receive the YUV planar image. Use
|
||
|
* {@link TJ#bufSizeYUV} to determine the appropriate size for this buffer
|
||
|
* based on the image width, height, and level of chrominance subsampling.
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*/
|
||
|
public void encodeYUV(BufferedImage srcImage, byte[] dstBuf, int flags)
|
||
|
throws Exception {
|
||
|
if (srcImage == null || dstBuf == null || flags < 0)
|
||
|
throw new Exception("Invalid argument in encodeYUV()");
|
||
|
int width = srcImage.getWidth();
|
||
|
int height = srcImage.getHeight();
|
||
|
int pixelFormat; boolean intPixels = false;
|
||
|
if (byteOrder == null)
|
||
|
byteOrder = ByteOrder.nativeOrder();
|
||
|
switch(srcImage.getType()) {
|
||
|
case BufferedImage.TYPE_3BYTE_BGR:
|
||
|
pixelFormat = TJ.PF_BGR; break;
|
||
|
case BufferedImage.TYPE_4BYTE_ABGR:
|
||
|
case BufferedImage.TYPE_4BYTE_ABGR_PRE:
|
||
|
pixelFormat = TJ.PF_XBGR; break;
|
||
|
case BufferedImage.TYPE_BYTE_GRAY:
|
||
|
pixelFormat = TJ.PF_GRAY; break;
|
||
|
case BufferedImage.TYPE_INT_BGR:
|
||
|
if (byteOrder == ByteOrder.BIG_ENDIAN)
|
||
|
pixelFormat = TJ.PF_XBGR;
|
||
|
else
|
||
|
pixelFormat = TJ.PF_RGBX;
|
||
|
intPixels = true; break;
|
||
|
case BufferedImage.TYPE_INT_RGB:
|
||
|
case BufferedImage.TYPE_INT_ARGB:
|
||
|
case BufferedImage.TYPE_INT_ARGB_PRE:
|
||
|
if (byteOrder == ByteOrder.BIG_ENDIAN)
|
||
|
pixelFormat = TJ.PF_XRGB;
|
||
|
else
|
||
|
pixelFormat = TJ.PF_BGRX;
|
||
|
intPixels = true; break;
|
||
|
default:
|
||
|
throw new Exception("Unsupported BufferedImage format");
|
||
|
}
|
||
|
WritableRaster wr = srcImage.getRaster();
|
||
|
if (subsamp < 0) throw new Exception("Subsampling level not set");
|
||
|
if (intPixels) {
|
||
|
SinglePixelPackedSampleModel sm =
|
||
|
(SinglePixelPackedSampleModel)srcImage.getSampleModel();
|
||
|
int stride = sm.getScanlineStride();
|
||
|
DataBufferInt db = (DataBufferInt)wr.getDataBuffer();
|
||
|
int[] buf = db.getData();
|
||
|
encodeYUV(buf, width, stride, height, pixelFormat, dstBuf, subsamp,
|
||
|
flags);
|
||
|
} else {
|
||
|
ComponentSampleModel sm =
|
||
|
(ComponentSampleModel)srcImage.getSampleModel();
|
||
|
int pixelSize = sm.getPixelStride();
|
||
|
if (pixelSize != TJ.getPixelSize(pixelFormat))
|
||
|
throw new Exception("Inconsistency between pixel format and pixel size in BufferedImage");
|
||
|
int pitch = sm.getScanlineStride();
|
||
|
DataBufferByte db = (DataBufferByte)wr.getDataBuffer();
|
||
|
byte[] buf = db.getData();
|
||
|
encodeYUV(buf, width, pitch, height, pixelFormat, dstBuf, subsamp,
|
||
|
flags);
|
||
|
}
|
||
|
compressedSize = TJ.bufSizeYUV(width, height, subsamp);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Encode the uncompressed source image stored in <code>srcImage</code>
|
||
|
* and return a buffer containing a YUV planar image. See
|
||
|
* {@link #encodeYUV(byte[], int)} for more detail.
|
||
|
*
|
||
|
* @param srcImage a <code>BufferedImage</code> instance containing RGB or
|
||
|
* grayscale pixels to be encoded
|
||
|
*
|
||
|
* @param flags the bitwise OR of one or more of
|
||
|
* {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
|
||
|
*
|
||
|
* @return a buffer containing a YUV planar image
|
||
|
*/
|
||
|
public byte[] encodeYUV(BufferedImage srcImage, int flags) throws Exception {
|
||
|
if (subsamp < 0)
|
||
|
throw new Exception("Subsampling level not set");
|
||
|
int width = srcImage.getWidth();
|
||
|
int height = srcImage.getHeight();
|
||
|
byte[] buf = new byte[TJ.bufSizeYUV(width, height, subsamp)];
|
||
|
encodeYUV(srcImage, buf, flags);
|
||
|
return buf;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the size of the image (in bytes) generated by the most recent
|
||
|
* compress/encode operation.
|
||
|
*
|
||
|
* @return the size of the image (in bytes) generated by the most recent
|
||
|
* compress/encode operation
|
||
|
*/
|
||
|
public int getCompressedSize() {
|
||
|
return compressedSize;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Free the native structures associated with this compressor instance.
|
||
|
*/
|
||
|
public void close() throws Exception {
|
||
|
destroy();
|
||
|
}
|
||
|
|
||
|
protected void finalize() throws Throwable {
|
||
|
try {
|
||
|
close();
|
||
|
} catch(Exception e) {
|
||
|
} finally {
|
||
|
super.finalize();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
private native void init() throws Exception;
|
||
|
|
||
|
private native void destroy() throws Exception;
|
||
|
|
||
|
// JPEG size in bytes is returned
|
||
|
private native int compress(byte[] srcBuf, int width, int pitch,
|
||
|
int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual,
|
||
|
int flags) throws Exception; // deprecated
|
||
|
|
||
|
private native int compress(byte[] srcBuf, int x, int y, int width,
|
||
|
int pitch, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp,
|
||
|
int jpegQual, int flags) throws Exception;
|
||
|
|
||
|
private native int compress(int[] srcBuf, int width, int stride,
|
||
|
int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual,
|
||
|
int flags) throws Exception; // deprecated
|
||
|
|
||
|
private native int compress(int[] srcBuf, int x, int y, int width,
|
||
|
int stride, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp,
|
||
|
int jpegQual, int flags) throws Exception;
|
||
|
|
||
|
private native void encodeYUV(byte[] srcBuf, int width, int pitch,
|
||
|
int height, int pixelFormat, byte[] dstBuf, int subsamp, int flags)
|
||
|
throws Exception;
|
||
|
|
||
|
private native void encodeYUV(int[] srcBuf, int width, int stride,
|
||
|
int height, int pixelFormat, byte[] dstBuf, int subsamp, int flags)
|
||
|
throws Exception;
|
||
|
|
||
|
static {
|
||
|
TJLoader.load();
|
||
|
}
|
||
|
|
||
|
private long handle = 0;
|
||
|
private byte[] srcBuf = null;
|
||
|
private int srcWidth = 0;
|
||
|
private int srcHeight = 0;
|
||
|
private int srcX = -1;
|
||
|
private int srcY = -1;
|
||
|
private int srcPitch = 0;
|
||
|
private int srcPixelFormat = -1;
|
||
|
private int subsamp = -1;
|
||
|
private int jpegQuality = -1;
|
||
|
private int compressedSize = 0;
|
||
|
private ByteOrder byteOrder = null;
|
||
|
};
|