DOWNLOAD
package Tiff; import Utils.BitReader;import Utils.ByteVectorOutputStream;import java.io.*;import java.net.*;import java.awt.image.*;import javax.imageio.ImageIO;import java.util.zip.Inflater;import java.util.zip.DataFormatException; // Modified version of ij.io.ImageReader /** Reads raw 8-bit, 16-bit or 32-bit (float or RGB) images from a stream or URL. */public class ImageReader { private static final int CLEAR_CODE = 256; private static final int EOI_CODE = 257; private ImageInfo fi; private int width, height; private long skipCount; private int bytesPerPixel, bufferSize, nPixels; private long byteCount; private int eofErrorCount; private int imageCount; private long startTime; public double min, max; // readRGB48() calculates min/max pixel values /** Constructs a new ImageReader using a ImageInfo object to describe the file to be read. @see ij.io.ImageInfo */ public ImageReader(ImageInfo fi) { this.fi = fi; width = fi.width; height = fi.height; skipCount = fi.getOffset(); } void eofError() { eofErrorCount++; } byte[] read8bitImage(InputStream in) throws IOException { if (fi.compression>ImageInfo.COMPRESSION_NONE) return readCompressed8bitImage(in); byte[] pixels = new byte[nPixels]; // assume contiguous strips int count, actuallyRead; int totalRead = 0; while (totalRead<byteCount) { if (totalRead+bufferSize>byteCount) count = (int)(byteCount-totalRead); else count = bufferSize; actuallyRead = in.read(pixels, totalRead, count); if (actuallyRead==-1) {eofError(); break;} totalRead += actuallyRead; } return pixels; } byte[] readCompressed8bitImage(InputStream in) throws IOException { byte[] pixels = new byte[nPixels]; int current = 0; byte last = 0; for (int i=0; i<fi.stripOffsets.length; i++) { /*if (in instanceof RandomAccessStream) ((RandomAccessStream)in).seek(fi.stripOffsets[i]); else */if (i > 0) { long skip = (fi.stripOffsets[i]&0xffffffffL) - (fi.stripOffsets[i-1]&0xffffffffL) - fi.stripLengths[i-1]; if (skip > 0L) in.skip(skip); } byte[] byteArray = new byte[fi.stripLengths[i]]; int read = 0, left = byteArray.length; while (left > 0) { int r = in.read(byteArray, read, left); if (r == -1) {eofError(); break;} read += r; left -= r; } byteArray = uncompress(byteArray); int length = byteArray.length; length = length - (length%fi.width); if (fi.compression==ImageInfo.LZW_WITH_DIFFERENCING||fi.compression==ImageInfo.ZIP_WITH_DIFFERENCING) { for (int b=0; b<length; b++) { byteArray[b] += last; last = b % fi.width == fi.width - 1 ? 0 : byteArray[b]; } } if (current+length>pixels.length) length = pixels.length-current; System.arraycopy(byteArray, 0, pixels, current, length); current += length; } return pixels; } /** Reads a 16-bit image. Signed pixels are converted to unsigned by adding 32768. */ short[] read16bitImage(InputStream in) throws IOException { if (fi.compression>ImageInfo.COMPRESSION_NONE || (fi.stripOffsets!=null&&fi.stripOffsets.length>1) && fi.fileType!=ImageInfo.RGB48_PLANAR) return readCompressed16bitImage(in); int pixelsRead; byte[] buffer = new byte[bufferSize]; short[] pixels = new short[nPixels]; long totalRead = 0L; int base = 0; int count, value; int bufferCount; while (totalRead<byteCount) { if ((totalRead+bufferSize)>byteCount) bufferSize = (int)(byteCount-totalRead); bufferCount = 0; while (bufferCount<bufferSize) { // fill the buffer count = in.read(buffer, bufferCount, bufferSize-bufferCount); if (count==-1) { for (int i=bufferCount; i<bufferSize; i++) buffer[i] = 0; totalRead = byteCount; eofError(); break; } bufferCount += count; } totalRead += bufferSize; pixelsRead = bufferSize/bytesPerPixel; if (fi.intelByteOrder) { if (fi.fileType==ImageInfo.GRAY16_SIGNED) for (int i=base,j=0; i<(base+pixelsRead); i++,j+=2) pixels[i] = (short)((((buffer[j+1]&0xff)<<8) | (buffer[j]&0xff))+32768); else for (int i=base,j=0; i<(base+pixelsRead); i++,j+=2) pixels[i] = (short)(((buffer[j+1]&0xff)<<8) | (buffer[j]&0xff)); } else { if (fi.fileType==ImageInfo.GRAY16_SIGNED) for (int i=base,j=0; i<(base+pixelsRead); i++,j+=2) pixels[i] = (short)((((buffer[j]&0xff)<<8) | (buffer[j+1]&0xff))+32768); else for (int i=base,j=0; i<(base+pixelsRead); i++,j+=2) pixels[i] = (short)(((buffer[j]&0xff)<<8) | (buffer[j+1]&0xff)); } base += pixelsRead; } return pixels; } short[] readCompressed16bitImage(InputStream in) throws IOException { //System.out.println("ImageReader.read16bit, offset="+fi.stripOffsets[0]); short[] pixels = new short[nPixels]; int base = 0; short last = 0; for (int k=0; k<fi.stripOffsets.length; k++) { //System.out.println("seek: "+k+" "+fi.stripOffsets[k]+" "+fi.stripLengths[k]+" "+(in instanceof RandomAccessStream)); /*if (in instanceof RandomAccessStream) ((RandomAccessStream)in).seek(fi.stripOffsets[k]); else */if (k > 0) { long skip = (fi.stripOffsets[k]&0xffffffffL) - (fi.stripOffsets[k-1]&0xffffffffL) - fi.stripLengths[k-1]; if (skip > 0L) in.skip(skip); } byte[] byteArray = new byte[fi.stripLengths[k]]; int read = 0, left = byteArray.length; while (left > 0) { int r = in.read(byteArray, read, left); if (r == -1) {eofError(); break;} read += r; left -= r; } byteArray = uncompress(byteArray); int pixelsRead = byteArray.length/bytesPerPixel; pixelsRead = pixelsRead - (pixelsRead%fi.width); int pmax = base+pixelsRead; if (pmax > nPixels) pmax = nPixels; if (fi.intelByteOrder) { for (int i=base,j=0; i<pmax; i++,j+=2) pixels[i] = (short)(((byteArray[j+1]&0xff)<<8) | (byteArray[j]&0xff)); } else { for (int i=base,j=0; i<pmax; i++,j+=2) pixels[i] = (short)(((byteArray[j]&0xff)<<8) | (byteArray[j+1]&0xff)); } if (fi.compression==ImageInfo.LZW_WITH_DIFFERENCING||fi.compression==ImageInfo.ZIP_WITH_DIFFERENCING) { for (int b=base; b<pmax; b++) { pixels[b] += last; last = b % fi.width == fi.width - 1 ? 0 : pixels[b]; } } base += pixelsRead; } if (fi.fileType==ImageInfo.GRAY16_SIGNED) { // convert to unsigned for (int i=0; i<nPixels; i++) pixels[i] = (short)(pixels[i]+32768); } return pixels; } float[] read32bitImage(InputStream in) throws IOException { if (fi.compression>ImageInfo.COMPRESSION_NONE || (fi.stripOffsets!=null&&fi.stripOffsets.length>1)) return readCompressed32bitImage(in); int pixelsRead; byte[] buffer = new byte[bufferSize]; float[] pixels = new float[nPixels]; long totalRead = 0L; int base = 0; int count, value; int bufferCount; int tmp; while (totalRead<byteCount) { if ((totalRead+bufferSize)>byteCount) bufferSize = (int)(byteCount-totalRead); bufferCount = 0; while (bufferCount<bufferSize) { // fill the buffer count = in.read(buffer, bufferCount, bufferSize-bufferCount); if (count==-1) { for (int i=bufferCount; i<bufferSize; i++) buffer[i] = 0; totalRead = byteCount; eofError(); break; } bufferCount += count; } totalRead += bufferSize; pixelsRead = bufferSize/bytesPerPixel; int pmax = base+pixelsRead; if (pmax>nPixels) pmax = nPixels; int j = 0; if (fi.intelByteOrder) for (int i=base; i<pmax; i++) { tmp = (int)(((buffer[j+3]&0xff)<<24) | ((buffer[j+2]&0xff)<<16) | ((buffer[j+1]&0xff)<<8) | (buffer[j]&0xff)); if (fi.fileType==ImageInfo.GRAY32_FLOAT) pixels[i] = Float.intBitsToFloat(tmp); else if (fi.fileType==ImageInfo.GRAY32_UNSIGNED) pixels[i] = (float)(tmp&0xffffffffL); else pixels[i] = tmp; j += 4; } else for (int i=base; i<pmax; i++) { tmp = (int)(((buffer[j]&0xff)<<24) | ((buffer[j+1]&0xff)<<16) | ((buffer[j+2]&0xff)<<8) | (buffer[j+3]&0xff)); if (fi.fileType==ImageInfo.GRAY32_FLOAT) pixels[i] = Float.intBitsToFloat(tmp); else if (fi.fileType==ImageInfo.GRAY32_UNSIGNED) pixels[i] = (float)(tmp&0xffffffffL); else pixels[i] = tmp; j += 4; } base += pixelsRead; } return pixels; } float[] readCompressed32bitImage(InputStream in) throws IOException { float[] pixels = new float[nPixels]; int base = 0; float last = 0; for (int k=0; k<fi.stripOffsets.length; k++) { /*if (in instanceof RandomAccessStream) ((RandomAccessStream)in).seek(fi.stripOffsets[k]); else */if (k > 0) { long skip = (fi.stripOffsets[k]&0xffffffffL) - (fi.stripOffsets[k-1]&0xffffffffL) - fi.stripLengths[k-1]; if (skip > 0L) in.skip(skip); } byte[] byteArray = new byte[fi.stripLengths[k]]; int read = 0, left = byteArray.length; while (left > 0) { int r = in.read(byteArray, read, left); if (r == -1) {eofError(); break;} read += r; left -= r; } byteArray = uncompress(byteArray); int pixelsRead = byteArray.length/bytesPerPixel; pixelsRead = pixelsRead - (pixelsRead%fi.width); int pmax = base+pixelsRead; if (pmax > nPixels) pmax = nPixels; int tmp; if (fi.intelByteOrder) { for (int i=base,j=0; i<pmax; i++,j+=4) { tmp = (int)(((byteArray[j+3]&0xff)<<24) | ((byteArray[j+2]&0xff)<<16) | ((byteArray[j+1]&0xff)<<8) | (byteArray[j]&0xff)); if (fi.fileType==ImageInfo.GRAY32_FLOAT) pixels[i] = Float.intBitsToFloat(tmp); else if (fi.fileType==ImageInfo.GRAY32_UNSIGNED) pixels[i] = (float)(tmp&0xffffffffL); else pixels[i] = tmp; } } else { for (int i=base,j=0; i<pmax; i++,j+=4) { tmp = (int)(((byteArray[j]&0xff)<<24) | ((byteArray[j+1]&0xff)<<16) | ((byteArray[j+2]&0xff)<<8) | (byteArray[j+3]&0xff)); if (fi.fileType==ImageInfo.GRAY32_FLOAT) pixels[i] = Float.intBitsToFloat(tmp); else if (fi.fileType==ImageInfo.GRAY32_UNSIGNED) pixels[i] = (float)(tmp&0xffffffffL); else pixels[i] = tmp; } } if (fi.compression==ImageInfo.LZW_WITH_DIFFERENCING||fi.compression==ImageInfo.ZIP_WITH_DIFFERENCING) { for (int b=base; b<pmax; b++) { pixels[b] += last; last = b % fi.width == fi.width - 1 ? 0 : pixels[b]; } } base += pixelsRead; } return pixels; } float[] read64bitImage(InputStream in) throws IOException { int pixelsRead; byte[] buffer = new byte[bufferSize]; float[] pixels = new float[nPixels]; long totalRead = 0L; int base = 0; int count, value; int bufferCount; long tmp; long b1, b2, b3, b4, b5, b6, b7, b8; while (totalRead<byteCount) { if ((totalRead+bufferSize)>byteCount) bufferSize = (int)(byteCount-totalRead); bufferCount = 0; while (bufferCount<bufferSize) { // fill the buffer count = in.read(buffer, bufferCount, bufferSize-bufferCount); if (count==-1) { for (int i=bufferCount; i<bufferSize; i++) buffer[i] = 0; totalRead = byteCount; eofError(); break; } bufferCount += count; } totalRead += bufferSize; pixelsRead = bufferSize/bytesPerPixel; int j = 0; for (int i=base; i < (base+pixelsRead); i++) { b1 = buffer[j+7]&0xff; b2 = buffer[j+6]&0xff; b3 = buffer[j+5]&0xff; b4 = buffer[j+4]&0xff; b5 = buffer[j+3]&0xff; b6 = buffer[j+2]&0xff; b7 = buffer[j+1]&0xff; b8 = buffer[j]&0xff; if (fi.intelByteOrder) tmp = (long)((b1<<56)|(b2<<48)|(b3<<40)|(b4<<32)|(b5<<24)|(b6<<16)|(b7<<8)|b8); else tmp = (long)((b8<<56)|(b7<<48)|(b6<<40)|(b5<<32)|(b4<<24)|(b3<<16)|(b2<<8)|b1); pixels[i] = (float)Double.longBitsToDouble(tmp); j += 8; } base += pixelsRead; } return pixels; } int[] readChunkyRGB(InputStream in, int[] pixels) throws IOException { if (fi.compression==ImageInfo.JPEG) return readJPEG(in, pixels); else if (fi.compression>ImageInfo.COMPRESSION_NONE || (fi.stripOffsets!=null&&fi.stripOffsets.length>1)) return readCompressedChunkyRGB(in, pixels); int pixelsRead; bufferSize = 24*width; byte[] buffer = new byte[bufferSize]; long totalRead = 0L; int base = 0; int count, value; int bufferCount; int r, g, b, a; if (pixels == null) pixels = new int[nPixels]; while (totalRead<byteCount) { if ((totalRead+bufferSize)>byteCount) bufferSize = (int)(byteCount-totalRead); bufferCount = 0; while (bufferCount<bufferSize) { // fill the buffer count = in.read(buffer, bufferCount, bufferSize-bufferCount); if (count==-1) { for (int i=bufferCount; i<bufferSize; i++) buffer[i] = 0; totalRead = byteCount; eofError(); break; } bufferCount += count; } totalRead += bufferSize; pixelsRead = bufferSize/bytesPerPixel; boolean bgr = fi.fileType==ImageInfo.BGR; int j = 0; for (int i=base; i<(base+pixelsRead); i++) { if (bytesPerPixel==4) { if (fi.fileType==ImageInfo.BARG) { // MCID b = buffer[j++]&0xff; j++; // ignore alfa byte r = buffer[j++]&0xff; g = buffer[j++]&0xff; } else if (fi.fileType==ImageInfo.ABGR) { b = buffer[j++]&0xff; g = buffer[j++]&0xff; r = buffer[j++]&0xff; j++; // ignore alfa byte } else if (fi.fileType==ImageInfo.CMYK) { r = buffer[j++] & 0xff; // c g = buffer[j++] & 0xff; // m b = buffer[j++] & 0xff; // y a = buffer[j++] & 0xff; // k if (a>0) { // if k>0 then c=c*(1-k)+k r = ((r*(256 - a))>>8) + a; g = ((g*(256 - a))>>8) + a; b = ((b*(256 - a))>>8) + a; } // else r=1-c, g=1-m and b=1-y, which IJ does by inverting image } else { // ARGB r = buffer[j++]&0xff; g = buffer[j++]&0xff; b = buffer[j++]&0xff; j++; // ignore alfa byte } } else { r = buffer[j++]&0xff; g = buffer[j++]&0xff; b = buffer[j++]&0xff; } if (bgr) pixels[i] = 0xff000000 | (b<<16) | (g<<8) | r; else pixels[i] = 0xff000000 | (r<<16) | (g<<8) | b; } base += pixelsRead; } return pixels; } int[] readCompressedChunkyRGB(InputStream in, int[] pixels) throws IOException { if (pixels == null) pixels = new int[nPixels]; int base = 0; int lastRed=0, lastGreen=0, lastBlue=0; int nextByte; int red=0, green=0, blue=0, alpha = 0; boolean bgr = fi.fileType==ImageInfo.BGR; boolean cmyk = fi.fileType==ImageInfo.CMYK; boolean differencing = fi.compression==ImageInfo.LZW_WITH_DIFFERENCING||fi.compression==ImageInfo.ZIP_WITH_DIFFERENCING; for (int i=0; i<fi.stripOffsets.length; i++) { /*if (in instanceof RandomAccessStream) ((RandomAccessStream)in).seek(fi.stripOffsets[i]); else */if (i > 0) { long skip = (fi.stripOffsets[i]&0xffffffffL) - (fi.stripOffsets[i-1]&0xffffffffL) - fi.stripLengths[i-1]; if (skip > 0L) in.skip(skip); } byte[] byteArray = new byte[fi.stripLengths[i]]; int read = 0, left = byteArray.length; while (left > 0) { int r = in.read(byteArray, read, left); if (r == -1) {eofError(); break;} read += r; left -= r; } byteArray = uncompress(byteArray); if (differencing) { for (int b=0; b<byteArray.length; b++) { if (b / bytesPerPixel % fi.width == 0) continue; byteArray[b] += byteArray[b - bytesPerPixel]; } } int k = 0; int pixelsRead = byteArray.length/bytesPerPixel; pixelsRead = pixelsRead - (pixelsRead%fi.width); int pmax = base+pixelsRead; if (pmax > nPixels) pmax = nPixels; for (int j=base; j<pmax; j++) { if (bytesPerPixel==4) { red = byteArray[k++]&0xff; green = byteArray[k++]&0xff; blue = byteArray[k++]&0xff; alpha = byteArray[k++]&0xff; if (cmyk && alpha>0) { red = ((red*(256-alpha))>>8) + alpha; green = ((green*(256-alpha))>>8) + alpha; blue = ((blue*(256-alpha))>>8) + alpha; } } else { red = byteArray[k++]&0xff; green = byteArray[k++]&0xff; blue = byteArray[k++]&0xff; } if (bgr) pixels[j] = 0xff000000 | (blue<<16) | (green<<8) | red; else pixels[j] = 0xff000000 | (red<<16) | (green<<8) | blue; } base += pixelsRead; } return pixels; } int[] readJPEG(InputStream in, int[] pixels) throws IOException { BufferedImage bi = ImageIO.read(in); if (bi == null) return null; DataBuffer buffer = bi.getRaster().getDataBuffer(); if (buffer instanceof DataBufferInt) return ((DataBufferInt)buffer).getData(); if (pixels == null) pixels = new int[width * height]; bi.getData().getSamples(0, 0, width, height, 0, pixels); return pixels; } int[] readPlanarRGB(InputStream in, int[] pixels) throws IOException { if (fi.compression>ImageInfo.COMPRESSION_NONE || (fi.stripOffsets!=null&&fi.stripOffsets.length>1)) return readCompressedPlanarRGBImage(in, pixels); DataInputStream dis = new DataInputStream(in); int planeSize = nPixels; // 1/3 image size byte[] buffer = new byte[planeSize]; int r, g, b; if (pixels == null) pixels = new int[nPixels]; startTime = 0L; dis.readFully(buffer); for (int i=0; i < planeSize; i++) { r = buffer[i]&0xff; pixels[i] = 0xff000000 | (r<<16); } dis.readFully(buffer); for (int i=0; i < planeSize; i++) { g = buffer[i]&0xff; pixels[i] |= g<<8; } dis.readFully(buffer); for (int i=0; i < planeSize; i++) { b = buffer[i]&0xff; pixels[i] |= b; } return pixels; } int[] readCompressedPlanarRGBImage(InputStream in, int[] pixels) throws IOException { if (pixels == null) pixels = new int[nPixels]; int r, g, b; nPixels *= 3; // read all 3 planes byte[] buffer = readCompressed8bitImage(in); nPixels /= 3; for (int i=0; i<nPixels; i++) { r = buffer[i]&0xff; pixels[i] = 0xff000000 | (r<<16); } for (int i=0; i<nPixels; i++) { g = buffer[nPixels+i]&0xff; pixels[i] |= g<<8; } for (int i=0; i<nPixels; i++) { b = buffer[nPixels*2+i]&0xff; pixels[i] |= b; } return pixels; } /* private void showProgress(int current, int last) { if (showProgressBar && (System.currentTimeMillis()-startTime)>500L) IJ.showProgress(current, last); } private void showProgress(long current, long last) { showProgress((int)(current/10L), (int)(last/10L)); } */ Object readRGB48(InputStream in) throws IOException { if (fi.compression>ImageInfo.COMPRESSION_NONE) return readCompressedRGB48(in); int channels = fi.samplesPerPixel; if (channels==1) channels=3; short[][] stack = new short[channels][nPixels]; DataInputStream dis = new DataInputStream(in); int pixel = 0; int min=65535, max=0; if (fi.stripLengths==null) { fi.stripLengths = new int[fi.stripOffsets.length]; fi.stripLengths[0] = width*height*bytesPerPixel; } for (int i=0; i<fi.stripOffsets.length; i++) { if (i>0) { long skip = (fi.stripOffsets[i]&0xffffffffL) - (fi.stripOffsets[i-1]&0xffffffffL) - fi.stripLengths[i-1]; if (skip>0L) dis.skip(skip); } int len = fi.stripLengths[i]; int bytesToGo = (nPixels-pixel)*channels*2; if (len>bytesToGo) len = bytesToGo; byte[] buffer = new byte[len]; dis.readFully(buffer); int value; int channel=0; boolean intel = fi.intelByteOrder; for (int base=0; base<len; base+=2) { if (intel) value = ((buffer[base+1]&0xff)<<8) | (buffer[base]&0xff); else value = ((buffer[base]&0xff)<<8) | (buffer[base+1]&0xff); if (value<min) min = value; if (value>max) max = value; stack[channel][pixel] = (short)(value); channel++; if (channel==channels) { channel = 0; pixel++; } } } this.min=min; this.max=max; return stack; } Object readCompressedRGB48(InputStream in) throws IOException { if (fi.compression==ImageInfo.LZW_WITH_DIFFERENCING||fi.compression==ImageInfo.ZIP_WITH_DIFFERENCING) throw new IOException("ImageJ cannot open 48-bit compressed TIFFs with predictors"); int channels = 3; short[][] stack = new short[channels][nPixels]; DataInputStream dis = new DataInputStream(in); int pixel = 0; int min=65535, max=0; for (int i=0; i<fi.stripOffsets.length; i++) { if (i>0) { long skip = (fi.stripOffsets[i]&0xffffffffL) - (fi.stripOffsets[i-1]&0xffffffffL) - fi.stripLengths[i-1]; if (skip>0L) dis.skip(skip); } int len = fi.stripLengths[i]; byte[] buffer = new byte[len]; dis.readFully(buffer); buffer = uncompress(buffer); len = buffer.length; if (len % 2 != 0) len--; int value; int channel=0; boolean intel = fi.intelByteOrder; for (int base=0; base<len && pixel<nPixels; base+=2) { if (intel) value = ((buffer[base+1]&0xff)<<8) | (buffer[base]&0xff); else value = ((buffer[base]&0xff)<<8) | (buffer[base+1]&0xff); if (value<min) min = value; if (value>max) max = value; stack[channel][pixel] = (short)(value); channel++; if (channel==channels) { channel = 0; pixel++; } } } this.min=min; this.max=max; return stack; } Object readRGB48Planar(InputStream in) throws IOException { int channels = fi.samplesPerPixel; if (channels==1) channels=3; Object[] stack = new Object[channels]; for (int i=0; i<channels; i++) stack[i] = read16bitImage(in); return stack; } /** * Reads in the contents of a 10-bit TIFF grayscale * image file and converts to a 16-bit image. * 5 bytes are read-in at a time and produces 4 pixels. * @param in * @return the pixels of the 10-bit image * @throws IOException */ short[] read10bitImage(InputStream in) throws IOException { int bytesPerLine = (int)(width*1.25); // there are 1.25 bytes of data for each pixel (5 bytes per 4 pixels) if ((width&1)==1) bytesPerLine++; // add 1 if odd byte[] buffer = new byte[bytesPerLine*height]; short[] pixels = new short[nPixels]; DataInputStream dis = new DataInputStream(in); dis.readFully(buffer); for (int y=0; y<height; y++) { int index1 = y*bytesPerLine; int index2 = y*width; int count = 0; while (count<width) { if (index1 + 4 < buffer.length) { final short B0 = (short) (buffer[index1] & 0xFF); final short B1 = (short) (buffer[index1 + 1] & 0xFF); final short B2 = (short) (buffer[index1 + 2] & 0xFF); final short B3 = (short) (buffer[index1 + 3] & 0xFF); final short B4 = (short) (buffer[index1 + 4] & 0xFF); short b0, b1, b2, b3; // Set pixel 1 b0 = (short) (B0 << 2); b1 = (short) (B1 >> 6); pixels[index2 + count] = (short) (b0 | b1); count++; // set pixel 2 b1 = (short) ((B1 & (0xFF >> 2)) << 4); b2 = (short) (B2 >> 4); pixels[index2 + count] = (short) (b1 | b2); count++; // set pixel 3 b2 = (short) ((B2 & (0xFF >> 4)) << 6); b3 = (short) (B3 >> 2); pixels[index2 + count] = (short) (b2 | b3); count++; // set pixel 4 b3 = (short) ((B3 & (0xFF >> 6)) << 8); pixels[index2 + count] = (short) (b3 | B4); count++; index1 += 5; } else break; } } return pixels; } short[] read12bitImage(InputStream in) throws IOException { int bytesPerLine = (int)(width*1.5); if ((width&1)==1) bytesPerLine++; // add 1 if odd byte[] buffer = new byte[bytesPerLine*height]; short[] pixels = new short[nPixels]; DataInputStream dis = new DataInputStream(in); dis.readFully(buffer); for (int y=0; y<height; y++) { int index1 = y*bytesPerLine; int index2 = y*width; int count = 0; while (count<width) { pixels[index2+count] = (short)(((buffer[index1]&0xff)*16) + ((buffer[index1+1]>>4)&0xf)); count++; if (count==width) break; pixels[index2+count] = (short)(((buffer[index1+1]&0xf)*256) + (buffer[index1+2]&0xff)); count++; index1+=3; } } return pixels; } float[] read24bitImage(InputStream in) throws IOException { byte[] buffer = new byte[width*3]; float[] pixels = new float[nPixels]; int b1, b2, b3; DataInputStream dis = new DataInputStream(in); for (int y=0; y<height; y++) { dis.readFully(buffer); int b = 0; for (int x=0; x<width; x++) { b1 = buffer[b++]&0xff; b2 = buffer[b++]&0xff; b3 = buffer[b++]&0xff; pixels[x+y*width] = (b3<<16) | (b2<<8) | b1; } } return pixels; } byte[] read1bitImage(InputStream in) throws IOException { if (fi.compression==ImageInfo.LZW) throw new IOException("ImageJ cannot open 1-bit LZW compressed TIFFs"); int scan=(int)Math.ceil(width/8.0); int len = scan*height; byte[] buffer = new byte[len]; byte[] pixels = new byte[nPixels]; DataInputStream dis = new DataInputStream(in); dis.readFully(buffer); int value1,value2, offset, index; for (int y=0; y<height; y++) { offset = y*scan; index = y*width; for (int x=0; x<scan; x++) { value1 = buffer[offset+x]&0xff; for (int i=7; i>=0; i--) { value2 = (value1&(1<<i))!=0?255:0; if (index<pixels.length) pixels[index++] = (byte)value2; } } } return pixels; } void skip(InputStream in) throws IOException { if (skipCount>0) { long bytesRead = 0; int skipAttempts = 0; long count; while (bytesRead<skipCount) { count = in.skip(skipCount-bytesRead); skipAttempts++; if (count==-1 || skipAttempts>5) break; bytesRead += count; } } byteCount = ((long)width)*height*bytesPerPixel; if (fi.fileType==ImageInfo.BITMAP) { int scan=width/8, pad = width%8; if (pad>0) scan++; byteCount = scan*height; } nPixels = width*height; bufferSize = (int)(byteCount/25L); if (bufferSize<8192) bufferSize = 8192; else bufferSize = (bufferSize/8192)*8192; } /** Reads the image from the InputStream and returns the pixel array (byte, short, int or float). Returns null if there was an IO exception. Does not close the InputStream. */ public Object readPixels(InputStream in, int[] existingRgbPixels) { Object pixels; startTime = System.currentTimeMillis(); try { switch (fi.fileType) { case ImageInfo.GRAY8: case ImageInfo.COLOR8: bytesPerPixel = 1; skip(in); pixels = (Object)read8bitImage(in); break; case ImageInfo.GRAY16_SIGNED: case ImageInfo.GRAY16_UNSIGNED: bytesPerPixel = 2; skip(in); pixels = (Object)read16bitImage(in); break; case ImageInfo.GRAY32_INT: case ImageInfo.GRAY32_UNSIGNED: case ImageInfo.GRAY32_FLOAT: bytesPerPixel = 4; skip(in); pixels = (Object)read32bitImage(in); break; case ImageInfo.GRAY64_FLOAT: bytesPerPixel = 8; skip(in); pixels = (Object)read64bitImage(in); break; case ImageInfo.RGB: case ImageInfo.BGR: case ImageInfo.ARGB: case ImageInfo.ABGR: case ImageInfo.BARG: case ImageInfo.CMYK: bytesPerPixel = fi.getBytesPerPixel(); skip(in); pixels = (Object)readChunkyRGB(in, existingRgbPixels); break; case ImageInfo.RGB_PLANAR: /* if (!(in instanceof RandomAccessStream) && fi.stripOffsets!=null && fi.stripOffsets.length>1) in = new RandomAccessStream(in); */ bytesPerPixel = 3; skip(in); pixels = (Object)readPlanarRGB(in, existingRgbPixels); break; case ImageInfo.BITMAP: bytesPerPixel = 1; skip(in); pixels = (Object)read1bitImage(in); break; case ImageInfo.RGB48: bytesPerPixel = 6; skip(in); pixels = (Object)readRGB48(in); break; case ImageInfo.RGB48_PLANAR: bytesPerPixel = 2; skip(in); pixels = (Object)readRGB48Planar(in); break; case ImageInfo.GRAY12_UNSIGNED: skip(in); short[] data = read12bitImage(in); pixels = (Object)data; break; case ImageInfo.GRAY10_UNSIGNED: skip(in); data = read10bitImage(in); pixels = (Object)data; break; case ImageInfo.GRAY24_UNSIGNED: skip(in); pixels = (Object)read24bitImage(in); break; default: pixels = null; } imageCount++; return pixels; } catch (IOException e) { //System.out.println(e); return null; } } /** Skips the specified number of bytes, then reads an image and returns the pixel array (byte, short, int or float). Returns null if there was an IO exception. Does not close the InputStream. */ public Object readPixels(InputStream in, long skipCount, int[] existingRgbPixels) { this.skipCount = skipCount; Object pixels = readPixels(in, existingRgbPixels); if (eofErrorCount>(imageCount==1?1:0)) return null; else return pixels; } /** Reads the image from a URL and returns the pixel array (byte, short, int or float). Returns null if there was an IO exception. */ public Object readPixels(String url, int[] existingRgbPixels) { java.net.URL theURL; InputStream is; try {theURL = new URL(url);} catch (MalformedURLException e) {System.out.println(e); return null;} try {is = theURL.openStream();} catch (IOException e) {System.out.println(e); return null;} return readPixels(is, existingRgbPixels); } private byte[] uncompress(byte[] input) { if (fi.compression==ImageInfo.PACK_BITS) return packBitsUncompress(input, fi.rowsPerStrip*fi.width*fi.getBytesPerPixel()); else if (fi.compression==ImageInfo.LZW || fi.compression==ImageInfo.LZW_WITH_DIFFERENCING) return lzwUncompress(input); else if (fi.compression==ImageInfo.ZIP || fi.compression==ImageInfo.ZIP_WITH_DIFFERENCING) return zipUncompress(input); else return input; } /** TIFF Adobe ZIP support contributed by Jason Newton. */ public byte[] zipUncompress(byte[] input) { ByteVectorOutputStream imageBuffer = new ByteVectorOutputStream(); byte[] buffer = new byte[1024]; Inflater decompressor = new Inflater(); decompressor.setInput(input); try { while(!decompressor.finished()) { int rlen = decompressor.inflate(buffer); imageBuffer.write(buffer, 0, rlen); } } catch(Exception e){ System.out.println(e); } decompressor.end(); return imageBuffer.copy(); } /** * Utility method for decoding an LZW-compressed image strip. * Adapted from the TIFF 6.0 Specification: * http://partners.adobe.com/asn/developer/pdfs/tn/TIFF6.pdf (page 61) * Author: Curtis Rueden (ctrueden at wisc.edu) */ public byte[] lzwUncompress(byte[] input) { if (input==null || input.length==0) return input; byte[][] symbolTable = new byte[16384][1]; int bitsToRead = 9; int nextSymbol = 258; int code; int oldCode = -1; ByteVectorOutputStream out = new ByteVectorOutputStream(8192); BitReader bb = new BitReader(input); byte[] byteBuffer1 = new byte[16]; byte[] byteBuffer2 = new byte[16]; while (out.size < byteCount) { code = bb.getBits(bitsToRead); if (code==EOI_CODE || code==-1) break; if (code==CLEAR_CODE) { // initialize symbol table for (int i = 0; i < 256; i++) symbolTable[i][0] = (byte)i; nextSymbol = 258; bitsToRead = 9; code = bb.getBits(bitsToRead); if (code==EOI_CODE || code==-1) break; out.add(symbolTable[code]); oldCode = code; } else { if (oldCode==-1) oldCode=0; if (code<nextSymbol) { // code is in table out.add(symbolTable[code]); // add string to table ByteVectorOutputStream symbol = new ByteVectorOutputStream(byteBuffer1); symbol.add(symbolTable[oldCode]); symbol.add(symbolTable[code][0]); symbolTable[nextSymbol] = symbol.copy(); //** oldCode = code; nextSymbol++; } else { // out of table ByteVectorOutputStream symbol = new ByteVectorOutputStream(byteBuffer2); symbol.add(symbolTable[oldCode]); symbol.add(symbolTable[oldCode][0]); byte[] outString = symbol.copy(); out.add(outString); symbolTable[nextSymbol] = outString; //** oldCode = code; nextSymbol++; } if (nextSymbol == 511) { bitsToRead = 10; } if (nextSymbol == 1023) { bitsToRead = 11; } if (nextSymbol == 2047) { bitsToRead = 12; } } } return out.copy(); } /** Based on the Bio-Formats PackbitsCodec written by Melissa Linkert. */ public byte[] packBitsUncompress(byte[] input, int expected) { if (expected==0) expected = Integer.MAX_VALUE; ByteVectorOutputStream output = new ByteVectorOutputStream(1024); int index = 0; while (output.size < expected && index < input.length) { byte n = input[index++]; if (n>=0) { // 0 <= n <= 127 byte[] b = new byte[n+1]; for (int i=0; i<n+1; i++) b[i] = input[index++]; output.add(b); b = null; } else if (n != -128) { // -127 <= n <= -1 int len = -n + 1; byte inp = input[index++]; for (int i=0; i<len; i++) output.add(inp); } } return output.copy(); } /* void debug(String label, InputStream in) { int offset = -1; if (in instanceof RandomAccessStream) { try { offset = ((RandomAccessStream)in).getFilePointer(); } catch(Exception e) {} } System.out.println(label+": debug: offset="+offset+", fi="+fi); } */}