1 /* 2 * Copyright (C)2011-2012, 2014 D. R. Commander. All Rights Reserved. 3 * 4 * Redistribution and use in source and binary forms, with or without 5 * modification, are permitted provided that the following conditions are met: 6 * 7 * - Redistributions of source code must retain the above copyright notice, 8 * this list of conditions and the following disclaimer. 9 * - Redistributions in binary form must reproduce the above copyright notice, 10 * this list of conditions and the following disclaimer in the documentation 11 * and/or other materials provided with the distribution. 12 * - Neither the name of the libjpeg-turbo Project nor the names of its 13 * contributors may be used to endorse or promote products derived from this 14 * software without specific prior written permission. 15 * 16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS", 17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE 20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 26 * POSSIBILITY OF SUCH DAMAGE. 27 */ 28 29 /* 30 * This program demonstrates how to compress and decompress JPEG files using 31 * the TurboJPEG JNI wrapper 32 */ 33 34 import java.io.*; 35 import java.awt.*; 36 import java.awt.image.*; 37 import java.nio.*; 38 import javax.imageio.*; 39 import javax.swing.*; 40 import org.libjpegturbo.turbojpeg.*; 41 42 public class TJExample implements TJCustomFilter { 43 44 public static final String classname = new TJExample().getClass().getName(); 45 usage()46 private static void usage() throws Exception { 47 System.out.println("\nUSAGE: java " + classname + " <Input file> <Output file> [options]\n"); 48 System.out.println("Input and output files can be any image format that the Java Image I/O"); 49 System.out.println("extensions understand. If either filename ends in a .jpg extension, then"); 50 System.out.println("TurboJPEG will be used to compress or decompress the file.\n"); 51 System.out.println("Options:\n"); 52 System.out.println("-scale M/N = if the input image is a JPEG file, scale the width/height of the"); 53 System.out.print(" output image by a factor of M/N (M/N = "); 54 for (int i = 0; i < sf.length; i++) { 55 System.out.print(sf[i].getNum() + "/" + sf[i].getDenom()); 56 if (sf.length == 2 && i != sf.length - 1) 57 System.out.print(" or "); 58 else if (sf.length > 2) { 59 if (i != sf.length - 1) 60 System.out.print(", "); 61 if (i == sf.length - 2) 62 System.out.print("or "); 63 } 64 } 65 System.out.println(")\n"); 66 System.out.println("-samp <444|422|420|gray> = If the output image is a JPEG file, this specifies"); 67 System.out.println(" the level of chrominance subsampling to use when"); 68 System.out.println(" recompressing it. Default is to use the same level"); 69 System.out.println(" of subsampling as the input, if the input is a JPEG"); 70 System.out.println(" file, or 4:4:4 otherwise.\n"); 71 System.out.println("-q <1-100> = If the output image is a JPEG file, this specifies the JPEG"); 72 System.out.println(" quality to use when recompressing it (default = 95).\n"); 73 System.out.println("-hflip, -vflip, -transpose, -transverse, -rot90, -rot180, -rot270 ="); 74 System.out.println(" If the input image is a JPEG file, perform the corresponding lossless"); 75 System.out.println(" transform prior to decompression (these options are mutually exclusive)\n"); 76 System.out.println("-grayscale = If the input image is a JPEG file, perform lossless grayscale"); 77 System.out.println(" conversion prior to decompression (can be combined with the other"); 78 System.out.println(" transforms above)\n"); 79 System.out.println("-crop X,Y,WxH = If the input image is a JPEG file, perform lossless cropping"); 80 System.out.println(" prior to decompression. X,Y specifies the upper left corner of the"); 81 System.out.println(" cropping region, and WxH specifies its width and height. X,Y must be"); 82 System.out.println(" evenly divible by the MCU block size (8x8 if the source image was"); 83 System.out.println(" compressed using no subsampling or grayscale, or 16x8 for 4:2:2 or 16x16"); 84 System.out.println(" for 4:2:0.)\n"); 85 System.out.println("-display = Display output image (Output file need not be specified in this"); 86 System.out.println(" case.)\n"); 87 System.out.println("-fastupsample = Use the fastest chrominance upsampling algorithm available in"); 88 System.out.println(" the underlying codec\n"); 89 System.out.println("-fastdct = Use the fastest DCT/IDCT algorithms available in the underlying"); 90 System.out.println(" codec\n"); 91 System.out.println("-accuratedct = Use the most accurate DCT/IDCT algorithms available in the"); 92 System.out.println(" underlying codec\n"); 93 System.exit(1); 94 } 95 96 private static final String[] sampName = { 97 "4:4:4", "4:2:2", "4:2:0", "Grayscale", "4:4:0" 98 }; 99 main(String[] argv)100 public static void main(String[] argv) { 101 102 BufferedImage img = null; 103 byte[] bmpBuf = null; 104 TJTransform xform = new TJTransform(); 105 int flags = 0; 106 107 try { 108 109 sf = TJ.getScalingFactors(); 110 111 if (argv.length < 2) { 112 usage(); 113 } 114 115 TJScalingFactor scaleFactor = new TJScalingFactor(1, 1); 116 String inFormat = "jpg", outFormat = "jpg"; 117 int outSubsamp = -1, outQual = 95; 118 boolean display = false; 119 120 if (argv.length > 1) { 121 for (int i = 1; i < argv.length; i++) { 122 if (argv[i].length() < 2) 123 continue; 124 if (argv[i].length() > 2 && 125 argv[i].substring(0, 3).equalsIgnoreCase("-sc")) { 126 int match = 0; 127 if (i < argv.length - 1) { 128 String[] scaleArg = argv[++i].split("/"); 129 if (scaleArg.length == 2) { 130 TJScalingFactor tempsf = 131 new TJScalingFactor(Integer.parseInt(scaleArg[0]), 132 Integer.parseInt(scaleArg[1])); 133 for (int j = 0; j < sf.length; j++) { 134 if (tempsf.equals(sf[j])) { 135 scaleFactor = sf[j]; 136 match = 1; 137 break; 138 } 139 } 140 } 141 } 142 if (match != 1) usage(); 143 } 144 if (argv[i].equalsIgnoreCase("-h") || argv[i].equalsIgnoreCase("-?")) 145 usage(); 146 if (argv[i].length() > 2 && 147 argv[i].substring(0, 3).equalsIgnoreCase("-sa")) { 148 if (i < argv.length - 1) { 149 i++; 150 if (argv[i].substring(0, 1).equalsIgnoreCase("g")) 151 outSubsamp = TJ.SAMP_GRAY; 152 else if (argv[i].equals("444")) 153 outSubsamp = TJ.SAMP_444; 154 else if (argv[i].equals("422")) 155 outSubsamp = TJ.SAMP_422; 156 else if (argv[i].equals("420")) 157 outSubsamp = TJ.SAMP_420; 158 else 159 usage(); 160 } else 161 usage(); 162 } 163 if (argv[i].substring(0, 2).equalsIgnoreCase("-q")) { 164 if (i < argv.length - 1) { 165 int qual = Integer.parseInt(argv[++i]); 166 if (qual >= 1 && qual <= 100) 167 outQual = qual; 168 else 169 usage(); 170 } else 171 usage(); 172 } 173 if (argv[i].substring(0, 2).equalsIgnoreCase("-g")) 174 xform.options |= TJTransform.OPT_GRAY; 175 if (argv[i].equalsIgnoreCase("-hflip")) 176 xform.op = TJTransform.OP_HFLIP; 177 if (argv[i].equalsIgnoreCase("-vflip")) 178 xform.op = TJTransform.OP_VFLIP; 179 if (argv[i].equalsIgnoreCase("-transpose")) 180 xform.op = TJTransform.OP_TRANSPOSE; 181 if (argv[i].equalsIgnoreCase("-transverse")) 182 xform.op = TJTransform.OP_TRANSVERSE; 183 if (argv[i].equalsIgnoreCase("-rot90")) 184 xform.op = TJTransform.OP_ROT90; 185 if (argv[i].equalsIgnoreCase("-rot180")) 186 xform.op = TJTransform.OP_ROT180; 187 if (argv[i].equalsIgnoreCase("-rot270")) 188 xform.op = TJTransform.OP_ROT270; 189 if (argv[i].equalsIgnoreCase("-custom")) 190 xform.cf = new TJExample(); 191 else if (argv[i].length() > 2 && 192 argv[i].substring(0, 2).equalsIgnoreCase("-c")) { 193 if (i >= argv.length - 1) 194 usage(); 195 String[] cropArg = argv[++i].split(","); 196 if (cropArg.length != 3) 197 usage(); 198 String[] dimArg = cropArg[2].split("[xX]"); 199 if (dimArg.length != 2) 200 usage(); 201 int tempx = Integer.parseInt(cropArg[0]); 202 int tempy = Integer.parseInt(cropArg[1]); 203 int tempw = Integer.parseInt(dimArg[0]); 204 int temph = Integer.parseInt(dimArg[1]); 205 if (tempx < 0 || tempy < 0 || tempw < 0 || temph < 0) 206 usage(); 207 xform.x = tempx; 208 xform.y = tempy; 209 xform.width = tempw; 210 xform.height = temph; 211 xform.options |= TJTransform.OPT_CROP; 212 } 213 if (argv[i].substring(0, 2).equalsIgnoreCase("-d")) 214 display = true; 215 if (argv[i].equalsIgnoreCase("-fastupsample")) { 216 System.out.println("Using fast upsampling code"); 217 flags |= TJ.FLAG_FASTUPSAMPLE; 218 } 219 if (argv[i].equalsIgnoreCase("-fastdct")) { 220 System.out.println("Using fastest DCT/IDCT algorithm"); 221 flags |= TJ.FLAG_FASTDCT; 222 } 223 if (argv[i].equalsIgnoreCase("-accuratedct")) { 224 System.out.println("Using most accurate DCT/IDCT algorithm"); 225 flags |= TJ.FLAG_ACCURATEDCT; 226 } 227 } 228 } 229 String[] inFileTokens = argv[0].split("\\."); 230 if (inFileTokens.length > 1) 231 inFormat = inFileTokens[inFileTokens.length - 1]; 232 String[] outFileTokens; 233 if (display) 234 outFormat = "bmp"; 235 else { 236 outFileTokens = argv[1].split("\\."); 237 if (outFileTokens.length > 1) 238 outFormat = outFileTokens[outFileTokens.length - 1]; 239 } 240 241 File file = new File(argv[0]); 242 int width, height; 243 244 if (inFormat.equalsIgnoreCase("jpg")) { 245 FileInputStream fis = new FileInputStream(file); 246 int inputSize = fis.available(); 247 if (inputSize < 1) { 248 System.out.println("Input file contains no data"); 249 System.exit(1); 250 } 251 byte[] inputBuf = new byte[inputSize]; 252 fis.read(inputBuf); 253 fis.close(); 254 255 TJDecompressor tjd; 256 if (xform.op != TJTransform.OP_NONE || xform.options != 0 || 257 xform.cf != null) { 258 TJTransformer tjt = new TJTransformer(inputBuf); 259 TJTransform[] t = new TJTransform[1]; 260 t[0] = xform; 261 t[0].options |= TJTransform.OPT_TRIM; 262 TJDecompressor[] tjdx = tjt.transform(t, 0); 263 tjd = tjdx[0]; 264 } else 265 tjd = new TJDecompressor(inputBuf); 266 267 width = tjd.getWidth(); 268 height = tjd.getHeight(); 269 int inSubsamp = tjd.getSubsamp(); 270 System.out.println("Source Image: " + width + " x " + height + 271 " pixels, " + sampName[inSubsamp] + " subsampling"); 272 if (outSubsamp < 0) 273 outSubsamp = inSubsamp; 274 275 if (outFormat.equalsIgnoreCase("jpg") && 276 (xform.op != TJTransform.OP_NONE || xform.options != 0) && 277 scaleFactor.isOne()) { 278 file = new File(argv[1]); 279 FileOutputStream fos = new FileOutputStream(file); 280 fos.write(tjd.getJPEGBuf(), 0, tjd.getJPEGSize()); 281 fos.close(); 282 System.exit(0); 283 } 284 285 width = scaleFactor.getScaled(width); 286 height = scaleFactor.getScaled(height); 287 288 if (!outFormat.equalsIgnoreCase("jpg")) 289 img = tjd.decompress(width, height, BufferedImage.TYPE_INT_RGB, 290 flags); 291 else 292 bmpBuf = tjd.decompress(width, 0, height, TJ.PF_BGRX, flags); 293 tjd.close(); 294 } else { 295 img = ImageIO.read(file); 296 if (img == null) 297 throw new Exception("Input image type not supported."); 298 width = img.getWidth(); 299 height = img.getHeight(); 300 if (outSubsamp < 0) { 301 if (img.getType() == BufferedImage.TYPE_BYTE_GRAY) 302 outSubsamp = TJ.SAMP_GRAY; 303 else 304 outSubsamp = TJ.SAMP_444; 305 } 306 } 307 System.gc(); 308 if (!display) 309 System.out.print("Dest. Image (" + outFormat + "): " + width + " x " + 310 height + " pixels"); 311 312 if (display) { 313 ImageIcon icon = new ImageIcon(img); 314 JLabel label = new JLabel(icon, JLabel.CENTER); 315 JOptionPane.showMessageDialog(null, label, "Output Image", 316 JOptionPane.PLAIN_MESSAGE); 317 } else if (outFormat.equalsIgnoreCase("jpg")) { 318 System.out.println(", " + sampName[outSubsamp] + 319 " subsampling, quality = " + outQual); 320 TJCompressor tjc = new TJCompressor(); 321 int jpegSize; 322 byte[] jpegBuf; 323 324 tjc.setSubsamp(outSubsamp); 325 tjc.setJPEGQuality(outQual); 326 if (img != null) 327 tjc.setSourceImage(img, 0, 0, 0, 0); 328 else { 329 tjc.setSourceImage(bmpBuf, 0, 0, width, 0, height, TJ.PF_BGRX); 330 } 331 jpegBuf = tjc.compress(flags); 332 jpegSize = tjc.getCompressedSize(); 333 tjc.close(); 334 335 file = new File(argv[1]); 336 FileOutputStream fos = new FileOutputStream(file); 337 fos.write(jpegBuf, 0, jpegSize); 338 fos.close(); 339 } else { 340 System.out.print("\n"); 341 file = new File(argv[1]); 342 ImageIO.write(img, outFormat, file); 343 } 344 345 } catch(Exception e) { 346 e.printStackTrace(); 347 System.exit(-1); 348 } 349 } 350 customFilter(ShortBuffer coeffBuffer, Rectangle bufferRegion, Rectangle planeRegion, int componentIndex, int transformIndex, TJTransform transform)351 public void customFilter(ShortBuffer coeffBuffer, Rectangle bufferRegion, 352 Rectangle planeRegion, int componentIndex, 353 int transformIndex, TJTransform transform) 354 throws Exception { 355 for (int i = 0; i < bufferRegion.width * bufferRegion.height; i++) { 356 coeffBuffer.put(i, (short)(-coeffBuffer.get(i))); 357 } 358 } 359 360 static TJScalingFactor[] sf = null; 361 }; 362