package com.nwalsh.saxon; import java.io.File; import java.io.BufferedReader; import java.io.FileReader; import java.io.IOException; import java.net.URL; import java.net.MalformedURLException; import java.awt.Toolkit; import java.awt.Image; import java.awt.image.ImageObserver; import java.lang.Thread; import java.util.StringTokenizer; /** *
Saxon extension to examine intrinsic size of images
* *$Id: ImageIntrinsics.java 5480 2005-12-06 18:56:52Z kosek $
* *Copyright (C) 2002 Norman Walsh.
* *This class provides a * Saxon * extension to find the intrinsic size of images.
* *Change Log:
*Initial release.
Constructor for ImageIntrinsics
*/ public ImageIntrinsics(String imageFn) { System.setProperty("java.awt.headless","true"); // Hack. I expect the right way to do this is to always use a URL. // However, that means getting the base URI correct and dealing // with the various permutations of the file: URI scheme on different // platforms. So instead, what we're going to do is cheat. If it // starts with http: or ftp:, call it a URI. Otherwise, call it // a file. Also call it a file if the URI is malformed. URL imageUrl = null; if (imageFn.startsWith("http:") || imageFn.startsWith("ftp:") || imageFn.startsWith("file:")) { try { imageUrl = new URL(imageFn); } catch (MalformedURLException mue) { imageUrl = null; } } if (imageUrl != null) { image = Toolkit.getDefaultToolkit().getImage (imageUrl); } else { image = Toolkit.getDefaultToolkit().getImage (imageFn); } width = image.getWidth(this); while (!imageFailed && (width == -1 || depth == -1)) { try { java.lang.Thread.currentThread().sleep(50); } catch (Exception e) { // nop; } width = image.getWidth(this); depth = image.getHeight(this); } if (imageFailed) { // Maybe it's an EPS or PDF? // FIXME: this code is crude (and doesn't handle the URL case!!!) BufferedReader ir = null; String line = null; int lineLimit = 100; try { ir = new BufferedReader(new FileReader(new File(imageFn))); line = ir.readLine(); if (line != null && line.startsWith("%PDF-")) { // We've got a PDF! while (lineLimit > 0 && line != null) { lineLimit--; if (line.startsWith("/CropBox [")) { line = line.substring(10); if (line.indexOf("]") >= 0) { line = line.substring(0, line.indexOf("]")); } parseBox(line); lineLimit = 0; } line = ir.readLine(); } } else if (line != null && line.startsWith("%!") && line.indexOf(" EPSF-") > 0) { // We've got an EPS! while (lineLimit > 0 && line != null) { lineLimit--; if (line.startsWith("%%BoundingBox: ")) { line = line.substring(15); parseBox(line); lineLimit = 0; } line = ir.readLine(); } } else { System.err.println("Failed to interpret image: " + imageFn); } } catch (Exception e) { System.err.println("Failed to load image: " + imageFn); } if (ir != null) { try { ir.close(); } catch (Exception e) { // nop; } } } } public int getWidth(int defaultWidth) { if (width >= 0) { return width; } else { return defaultWidth; } } public int getDepth(int defaultDepth) { if (depth >= 0) { return depth; } else { return defaultDepth; } } private void parseBox(String line) { int [] corners = new int [4]; int count = 0; StringTokenizer st = new StringTokenizer(line); while (count < 4 && st.hasMoreTokens()) { try { corners[count++] = Integer.parseInt(st.nextToken()); } catch (Exception e) { // nop; } } width = corners[2] - corners[0]; depth = corners[3] - corners[1]; } public boolean imageUpdate(Image img, int infoflags, int x, int y, int width, int height) { if ((infoflags & ImageObserver.ERROR) == ImageObserver.ERROR) { imageFailed = true; return false; } // I really only care about the width and height, but if I return false as // soon as those are available, the BufferedInputStream behind the loader // gets closed too early. int flags = ImageObserver.ALLBITS; if ((infoflags & flags) == flags) { return false; } else { return true; } } }