All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.twelvemonkeys.imageio.plugins.pict.PICTImageReader Maven / Gradle / Ivy

There is a newer version: 3.12.0
Show newest version
/*
 * Copyright (c) 2008, Harald Kuhr
 * 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 copyright holder 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 HOLDER 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.
 *
Parts of this software is based on JVG/JIS.
See http://www.cs.hut.fi/~framling/JVG/index.html for more information.
Redistribution under BSD authorized by Kary Främling:

Copyright (c) 2003, Kary Främling
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 JIS/JVG 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 OWNER 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 com.twelvemonkeys.imageio.plugins.pict;

import com.twelvemonkeys.imageio.ImageReaderBase;
import com.twelvemonkeys.imageio.util.IIOUtil;
import com.twelvemonkeys.imageio.util.ImageTypeSpecifiers;
import com.twelvemonkeys.io.enc.Decoder;
import com.twelvemonkeys.io.enc.DecoderStream;
import com.twelvemonkeys.io.enc.PackBitsDecoder;

import javax.imageio.IIOException;
import javax.imageio.ImageIO;
import javax.imageio.ImageReadParam;
import javax.imageio.ImageReader;
import javax.imageio.ImageTypeSpecifier;
import javax.imageio.metadata.IIOMetadata;
import javax.imageio.spi.ImageReaderSpi;
import javax.imageio.stream.ImageInputStream;
import java.awt.*;
import java.awt.color.*;
import java.awt.geom.*;
import java.awt.image.*;
import java.io.DataInput;
import java.io.DataInputStream;
import java.io.EOFException;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;

/**
 * Reader for Apple Mac Paint Picture (PICT) format.
 *
 * @author Harald Kuhr
 * @author Kary Främling (original PICT/QuickDraw parsing)
 * @author Matthias Wiesmann (original embedded QuickTime parsing)
 * @version $Id: PICTReader.java,v 1.0 05.apr.2006 15:20:48 haku Exp$
 */
/*
 * TODO: New paint strategy: Need to have a PEN and a PEN MODE, in addition to BG and PATTERN and PATTERN MODE.
 *       - These must be set before each frame/paint/invert/erase/fill operation.
 *         This is because there isn't a one-to-one mapping, between Java and PICT drawing.
 *        - Subclass Graphics?
 *        - Or create a QuickDrawContext that converts from PICT to AWT Graphics?
 *        - Or Methods like setupFrame(pen, penmode, penwidth?), setupPaint(pen, penmode), setupFill(patter, patMode), etc?
 *        - Or methods like frameRect(pen, penmode, penwidth, rect), frameOval(pen, penmode, penwidth, rect), etc?
 *        - Or methods like frameShape(pen, penmode, penwidth, shape), paintShape(pen, penmode, shape) etc??
 *      QuickDrawContext that wraps an AWT Grpahics, and with methods macthing opcodes, seems like the best fit ATM
 * TODO: Some MAJOR clean up
 * TODO: As we now have Graphics2D with more options, support more of the format?
 */
public final class PICTImageReader extends ImageReaderBase {

    final static boolean DEBUG = "true".equalsIgnoreCase(System.getProperty("com.twelvemonkeys.imageio.plugins.pict.debug"));

    private QuickDrawContext context;
    private Rectangle frame;

    // TODO: Do we need this?
    private int version;

    // Variables for storing draw status
    // TODO: Get rid of these, or move to context
    private Point penPosition = new Point(0, 0);
    private Rectangle lastRectangle = new Rectangle(0, 0);

    // Ratio between the screen resolution and the image resolution
    private double screenImageXRatio;
    private double screenImageYRatio;

    // List of images created during image import
    private final List images = new ArrayList<>();
    private long imageStartStreamPos;
    protected int picSize;

    @Deprecated
    public PICTImageReader() {
        this(null);
    }

    protected PICTImageReader(final ImageReaderSpi pProvider) {
        super(pProvider);
    }

    protected void resetMembers() {
        context = null;
        frame = null;
        images.clear();
    }

    /**
     * Open and read the frame size of the PICT file.
     *
     * @return return the PICT frame
     * @throws IOException if an I/O error occurs while reading the image.
     */
    private Rectangle getPICTFrame() throws IOException {
        if (frame == null) {
            // Read in header information
            readPICTHeader(imageInput);

            if (DEBUG) {
                System.out.println("Done reading PICT header!");
            }
        }

        return frame;
    }

    /**
     * Read the PICT header. The information read is shown on stdout if "DEBUG" is true.
     *
     * @param pStream the stream to read from
     *
     * @throws IOException if an I/O error occurs while reading the image.
     */
    private void readPICTHeader(final ImageInputStream pStream) throws IOException {
        pStream.seek(0L);

        try {
            readPICTHeader0(pStream);
        }
        catch (IIOException e) {
            // Rest and try again
            pStream.seek(0L);

            // Skip first 512 bytes
            PICTImageReaderSpi.skipNullHeader(pStream);
            readPICTHeader0(pStream);
        }
    }

    private void readPICTHeader0(final ImageInputStream pStream) throws IOException {
        // Get size
        picSize = pStream.readUnsignedShort();

        if (DEBUG) {
            System.out.println("picSize: " + picSize);
        }

        // Get frame at 72 dpi
        // NOTE: These are not pixel sizes!
        // Need to be multiplied with hRes/screenResolution and vRes/screenResolution
        int y = pStream.readUnsignedShort();
        int x = pStream.readUnsignedShort();
        int h = pStream.readUnsignedShort();
        int w = pStream.readUnsignedShort();

        frame = new Rectangle(x, y, w - x, h - y);
        if (frame.width < 0 || frame.height < 0) {
            throw new IIOException("Error in PICT header: Invalid frame " + frame);
        }
        if (DEBUG) {
            System.out.println("frame: " + frame);
        }

        // Set default display ratios. 72 dpi is the standard Mac resolution.
        screenImageXRatio = 1.0;
        screenImageYRatio = 1.0;

        // Get the version, since the way of reading the rest depends on it
        boolean isExtendedV2 = false;
        int version = pStream.readShort();
        if (DEBUG) {
            System.out.printf("PICT version: 0x%04x%n", version);
        }

        if (version == (PICT.OP_VERSION << 8) + 0x01) {
            this.version = 1;
        }
        else if (version == PICT.OP_VERSION && pStream.readShort() == PICT.OP_VERSION_2) {
            this.version = 2;

            // Read in version 2 header op and test that it is valid: HeaderOp 0x0C00
            if (pStream.readShort() != PICT.OP_HEADER_OP) {
                throw new IIOException("Error in PICT header: Invalid HeaderOp, expected 0x0c00");
            }

            int headerVersion = pStream.readInt();
            if (DEBUG) {
                System.out.printf("headerVersion: 0x%04x%n", headerVersion);
            }

            // TODO: This (headerVersion) should be picture size (bytes) for non-V2-EXT...?
            //       - but.. We should take care to make sure we don't mis-interpret non-PICT data...
            if ((headerVersion & 0xffff0000) != PICT.HEADER_V2_EXT) {
                // TODO: Test this.. Looks dodgy to me..
                // Get the image resolution and calculate the ratio between
                // the default Mac screen resolution and the image resolution

                // int y, x, w(?), h (fixed point)
                double y2 = PICTUtil.readFixedPoint(pStream);
                double x2 = PICTUtil.readFixedPoint(pStream);
                double w2 = PICTUtil.readFixedPoint(pStream); // ?!
                double h2 = PICTUtil.readFixedPoint(pStream);

                screenImageXRatio = (w - x) / (w2 - x2);
                screenImageYRatio = (h - y) / (h2 - y2);

                if (screenImageXRatio < 0 || screenImageYRatio < 0) {
                    throw new IIOException("Error in PICT header: Invalid bounds " + new Rectangle.Double(x2, y2, w2 - x2, h2 - y2));
                }
                if (DEBUG) {
                    System.out.println("bounding rect: " + new Rectangle.Double(x2, y2, w2 - x2, h2 - y2));
                }

                // int reserved
                pStream.skipBytes(4);
            }
            else {
                isExtendedV2 = true;
                // Get the image resolution
                // Not sure if they are useful for anything...

                // int horizontal res (fixed point)
                double xRes = PICTUtil.readFixedPoint(pStream);
                // int vertical res (fixed point)
                double yRes = PICTUtil.readFixedPoint(pStream);

                if (DEBUG) {
                    System.out.println("xResolution: " + xRes);
                    System.out.println("yResolution: " + yRes);
                }

                // Get the image resolution and calculate the ratio between
                // the default Mac screen resolution and the image resolution
                // short y, x, h, w
                short y2 = pStream.readShort();
                short x2 = pStream.readShort();
                short h2 = pStream.readShort();
                short w2 = pStream.readShort();

                screenImageXRatio = (w - x) / (double) (w2 - x2);
                screenImageYRatio = (h - y) / (double) (h2 - y2);

                if (screenImageXRatio < 0 || screenImageYRatio < 0) {
                    throw new IIOException("Error in PICT header: Invalid bounds " + new Rectangle.Double(x2, y2, w2 - x2, h2 - y2));
                }
                if (DEBUG) {
                    System.out.println("bounding rect: " + new Rectangle(x2, y2, w2 - x2, h2 - y2));
                }

                // long reserved
                pStream.skipBytes(4);
            }

            if (DEBUG) {
                System.out.println("screenImageXRatio: " + screenImageXRatio);
                System.out.println("screenImageYRatio: " + screenImageYRatio);
            }
        }
        else {
            // No version information, return straight away
            throw new IIOException("Error in PICT header: Missing or unknown version information");
        }

        if (DEBUG) {
            System.out.println("Version: " + this.version + (isExtendedV2 ? " extended" : ""));
        }

        imageStartStreamPos = pStream.getStreamPosition();

        // Won't need header data again (NOTE: We'll only get here if no exception is thrown)
        pStream.flushBefore(imageStartStreamPos);
    }

    /**
     * Reads the PICT stream.
     * The contents of the stream will be drawn onto the supplied graphics
     * object.
     * 

* If "DEBUG" is true, the elements read are listed on stdout. *

* * @param pGraphics the graphics object to draw onto. * * @throws IIOException if the data can not be read. * @throws IOException if an I/O error occurs while reading the image. */ private void drawOnto(Graphics2D pGraphics) throws IOException { context = new QuickDrawContext(pGraphics); readPICTopcodes(imageInput); if (DEBUG) { System.out.println("Done reading PICT body!"); } } /** * Parse PICT opcodes in a PICT file. The input stream must be * positioned at the beginning of the opcodes, after picframe. * If we have a non-null graphics, we try to draw the elements. * * @param pStream the stream to read from * * @throws IIOException if the data can not be read. * @throws IOException if an I/O error occurs while reading the image. */ private void readPICTopcodes(ImageInputStream pStream) throws IOException { pStream.seek(imageStartStreamPos); int opCode, dh, dv, dataLength; byte[] colorBuffer = new byte[3 * PICT.COLOR_COMP_SIZE]; Pattern fill = QuickDraw.BLACK; Pattern bg; Pattern pen; Color hilight = Color.RED; Point origin, dh_dv; Point ovSize = new Point(); Point arcAngles = new Point(); String text; Rectangle bounds = new Rectangle(); Polygon polygon = new Polygon(); Area region = new Area(); int pixmapCount = 0; try { // Read from file until we read the end of picture opcode do { // Read opcode, version 1: byte, version 2: short if (version == 1) { opCode = pStream.readUnsignedByte(); } else { // Always word-aligned for version 2 if ((pStream.getStreamPosition() & 1) > 0) { pStream.readByte(); } opCode = pStream.readUnsignedShort(); } // See what we got and react in consequence switch (opCode) { case PICT.NOP: // Just go on if (DEBUG) { System.out.println("NOP"); } break; case PICT.OP_CLIP_RGN:// OK for RECTs, not for regions yet // Read the region if ((region = readRegion(pStream, bounds)) == null) { throw new IIOException("Could not read region"); } // Set clip rect or clip region //if (mGraphics != null) { // if (region.npoints == 0) { // // TODO: Read what the specs says about this... // if (bounds.width > 0 && bounds.height > 0) { // mGraphics.setClip(bounds.x, bounds.y, bounds.width, bounds.height); // } // } // else { // mGraphics.setClip(region); // } //} if (DEBUG) { verboseRegionCmd("clipRgn", bounds, region); } break; case PICT.OP_BK_PAT: // Get the data context.setBackgroundPattern(PICTUtil.readPattern(pStream)); if (DEBUG) { System.out.println("bkPat"); } break; case PICT.OP_TX_FONT: // Get the data byte[] fontData = new byte[2]; pStream.readFully(fontData, 0, 2); // TODO: Font family id, 0 - System font, 1 - Application font. // But how can we get these mappings? if (DEBUG) { System.out.println("txFont: " + Arrays.toString(fontData)); } break; case PICT.OP_TX_FACE: // Get the data int txFace = pStream.readUnsignedByte(); context.setTextFace(txFace); if (DEBUG) { System.out.println("txFace: " + txFace); } break; case PICT.OP_TX_MODE: // Get the data byte[] mode_buf = new byte[2]; pStream.readFully(mode_buf, 0, mode_buf.length); if (DEBUG) { System.out.println("txMode: " + mode_buf[0] + ", " + mode_buf[1]); } break; case PICT.OP_SP_EXTRA:// WONDER WHAT IT IS? // Get the data pStream.readFully(new byte[4], 0, 4); if (DEBUG) { System.out.println("spExtra"); } break; case PICT.OP_PN_SIZE: // Get the two words // NOTE: This is out of order, compared to other Points Dimension pnsize = new Dimension(pStream.readUnsignedShort(), pStream.readUnsignedShort()); if (DEBUG) { System.out.println("pnsize: " + pnsize); } context.setPenSize(pnsize); break; case PICT.OP_PN_MODE: // Get the data int mode = pStream.readUnsignedShort(); if (DEBUG) { System.out.println("pnMode: " + mode); } context.setPenMode(mode); break; case PICT.OP_PN_PAT: context.setPenPattern(PICTUtil.readPattern(pStream, context.getForeground(), context.getBackground())); if (DEBUG) { System.out.println("pnPat"); } break; case PICT.OP_FILL_PAT: fill = PICTUtil.readPattern(pStream); if (DEBUG) { System.out.println("fillPat"); } break; case PICT.OP_OV_SIZE:// OK, we use this for rounded rectangle corners // Get the two words int y = getYPtCoord(pStream.readUnsignedShort()); int x = getXPtCoord(pStream.readUnsignedShort()); ovSize.setLocation(x, y); /* ovSize.x *= 2;// Don't know why, but has to be multiplied by 2 ovSize.y *= 2; */ if (DEBUG) { System.out.println("ovSize: " + ovSize); } break; case PICT.OP_ORIGIN:// PROBABLY OK // Get the two words y = getYPtCoord(pStream.readUnsignedShort()); x = getXPtCoord(pStream.readUnsignedShort()); origin = new Point(x, y); if (DEBUG) { System.out.println("Origin: " + origin); } break; case PICT.OP_TX_SIZE:// OK // Get the text size int tx_size = getYPtCoord(pStream.readUnsignedShort()); context.setTextSize(tx_size); if (DEBUG) { System.out.println("txSize: " + tx_size); } break; case PICT.OP_FG_COLOR:// TO BE DONE IF POSSIBLE // TODO! // Get the data pStream.readInt(); if (DEBUG) { System.out.println("fgColor"); } break; case PICT.OP_BK_COLOR:// TO BE DONE IF POSSIBLE // TODO! // Get the data pStream.readInt(); if (DEBUG) { System.out.println("bgColor"); } break; case PICT.OP_TX_RATIO:// SEE IF WE HAVE THIS??? // Get the data pStream.readFully(new byte[8], 0, 8); if (DEBUG) { System.out.println("txRatio"); } break; case PICT.OP_VERSION:// OK, ignored since we should already have it // Get the data pStream.readFully(new byte[1], 0, 1); if (DEBUG) { System.out.println("opVersion"); } break; case 0x0012: // BkPixPat bg = PICTUtil.readColorPattern(pStream); context.setBackgroundPattern(bg); if (DEBUG) { System.out.println("BkPixPat"); } break; case 0x0013: // PnPixPat pen = PICTUtil.readColorPattern(pStream); context.setPenPattern(pen); if (DEBUG) { System.out.println("PnPixPat"); } break; case 0x0014: // FillPixPat fill = PICTUtil.readColorPattern(pStream); context.setFillPattern(fill); if (DEBUG) { System.out.println("FillPixPat"); } break; case PICT.OP_PN_LOC_H_FRAC:// TO BE DONE??? // Get the data pStream.readFully(new byte[2], 0, 2); if (DEBUG) { System.out.println("opPnLocHFrac"); } break; case PICT.OP_CH_EXTRA:// TO BE DONE??? // Get the data pStream.readFully(new byte[2], 0, 2); if (DEBUG) { System.out.println("opChExtra"); } break; case PICT.OP_RGB_FG_COL:// OK // Get the color pStream.readFully(colorBuffer, 0, colorBuffer.length); Color foreground = new Color((colorBuffer[0] & 0xFF), (colorBuffer[2] & 0xFF), (colorBuffer[4] & 0xFF)); if (DEBUG) { System.out.println("rgbFgColor: " + foreground); } context.setForeground(foreground); break; case PICT.OP_RGB_BK_COL:// OK // Get the color pStream.readFully(colorBuffer, 0, colorBuffer.length); // The color might be 16 bit per component.. Color background = new Color(colorBuffer[0] & 0xFF, colorBuffer[2] & 0xFF, colorBuffer[4] & 0xFF); if (DEBUG) { System.out.println("rgbBgColor: " + background); } context.setBackground(background); break; case PICT.OP_HILITE_MODE: // Change color to hilite color context.setPenPattern(new BitMapPattern(hilight)); if (DEBUG) { System.out.println("opHiliteMode"); } break; case PICT.OP_HILITE_COLOR:// OK // Get the color pStream.readFully(colorBuffer, 0, colorBuffer.length); // TODO: The color might be 16 bit per component.. hilight = new Color((colorBuffer[0] & 0xFF), (colorBuffer[2] & 0xFF), (colorBuffer[4] & 0xFF)); if (DEBUG) { System.out.println("opHiliteColor: " + hilight); } break; case PICT.OP_DEF_HILITE:// Macintosh internal, ignored? // Nothing to do hilight = Color.red; // TODO: My guess it's a reset, verify! if (DEBUG) { System.out.println("opDefHilite"); } break; case PICT.OP_OP_COLOR:// To be done once I know what it means // TODO: Is this the mask? Scale value for RGB colors? // Get the color pStream.readFully(colorBuffer, 0, colorBuffer.length); if (DEBUG) { System.out.println("opOpColor"); } break; case PICT.OP_LINE:// OK, not tested // Get the data (two points) y = getYPtCoord(pStream.readUnsignedShort()); x = getXPtCoord(pStream.readUnsignedShort()); origin = new Point(x, y); y = getYPtCoord(pStream.readUnsignedShort()); x = getXPtCoord(pStream.readUnsignedShort()); penPosition.setLocation(x, y); // Move pen to new position, draw line context.moveTo(origin); context.lineTo(penPosition); if (DEBUG) { System.out.println("line from: " + origin + " to: " + penPosition); } break; case PICT.OP_LINE_FROM:// OK, not tested // Get the point y = getYPtCoord(pStream.readUnsignedShort()); x = getXPtCoord(pStream.readUnsignedShort()); // Draw line context.line(x, y); if (DEBUG) { System.out.println("lineFrom to: " + penPosition); } break; case PICT.OP_SHORT_LINE:// OK // Get origin and dh, dv y = getYPtCoord(pStream.readUnsignedShort()); x = getXPtCoord(pStream.readUnsignedShort()); origin = new Point(x, y); x = getXPtCoord(pStream.readByte()); y = getYPtCoord(pStream.readByte()); dh_dv = new Point(x, y); // Move pen to new position, draw line if we have a graphics context.moveTo(origin); penPosition.setLocation(origin.x + dh_dv.x, origin.y + dh_dv.y); context.lineTo(penPosition); if (DEBUG) { System.out.println("Short line origin: " + origin + ", dh,dv: " + dh_dv); } break; case PICT.OP_SHORT_LINE_FROM:// OK // Get dh, dv x = getXPtCoord(pStream.readByte()); y = getYPtCoord(pStream.readByte()); // Draw line context.line(x, y); if (DEBUG) { System.out.println("Short line from dh,dv: " + x + "," + y); } break; case 0x24: case 0x25: case 0x26: case 0x27: case 0x2F: // Apple reserved dataLength = pStream.readUnsignedShort(); pStream.readFully(new byte[dataLength], 0, dataLength); if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; case PICT.OP_LONG_TEXT:// OK // Get the data y = getYPtCoord(pStream.readUnsignedShort()); x = getXPtCoord(pStream.readUnsignedShort()); origin = new Point(x, y); penPosition = origin; context.moveTo(penPosition); text = PICTUtil.readPascalString(pStream); context.drawString(text); if (DEBUG) { System.out.println("longText origin: " + penPosition + ", text:" + text); } break; case PICT.OP_DH_TEXT:// OK, not tested // Get dh dh = getXPtCoord(pStream.readByte()); penPosition.translate(dh, 0); context.moveTo(penPosition); text = PICTUtil.readPascalString(pStream); context.drawString(text); if (DEBUG) { System.out.println("DHText dh: " + dh + ", text:" + text); } break; case PICT.OP_DV_TEXT:// OK, not tested // Get dh dv = getYPtCoord(pStream.readByte()); penPosition.translate(0, dv); context.moveTo(penPosition); text = PICTUtil.readPascalString(pStream); context.drawString(text); if (DEBUG) { System.out.println("DVText dv: " + dv + ", text:" + text); } break; case PICT.OP_DHDV_TEXT:// OK, not tested // Get dh, dv y = getYPtCoord(pStream.readByte()); x = getXPtCoord(pStream.readByte()); penPosition.translate(x, y); context.moveTo(penPosition); text = PICTUtil.readPascalString(pStream); context.drawString(text); if (DEBUG) { System.out.println("DHDVText penPosition: " + penPosition + ", text:" + text); } break; case PICT.OP_FONT_NAME:// OK, not tested // Get data length /*data_len = */ pStream.readShort(); // Get old font ID, ignored pStream.readUnsignedShort(); // Get font name and set the new font if we have one String fontName = PICTUtil.readPascalString(pStream); context.setTextFont(fontName); if (DEBUG) { System.out.println("fontName: \"" + fontName +"\""); } break; case PICT.OP_LINE_JUSTIFY:// TODO // Get data byte[] lineJustifyData = new byte[10]; pStream.readFully(lineJustifyData, 0, lineJustifyData.length); if (DEBUG) { System.out.println("opLineJustify"); } break; case PICT.OP_GLYPH_STATE:// TODO: NOT SUPPORTED IN AWT GRAPHICS YET? // Get data byte[] glyphState = new byte[6]; pStream.readFully(glyphState, 0, glyphState.length); if (DEBUG) { System.out.println("glyphState: " + Arrays.toString(glyphState)); } break; //-------------------------------------------------------------------------------- // Rect treatments //-------------------------------------------------------------------------------- case PICT.OP_FRAME_RECT:// OK case PICT.OP_PAINT_RECT:// OK case PICT.OP_ERASE_RECT:// OK, not tested case PICT.OP_INVERT_RECT:// OK, not tested case PICT.OP_FILL_RECT:// OK, not tested // Get the frame rectangle readRectangle(pStream, lastRectangle); case PICT.OP_FRAME_SAME_RECT:// OK, not tested case PICT.OP_PAINT_SAME_RECT:// OK, not tested case PICT.OP_ERASE_SAME_RECT:// OK, not tested case PICT.OP_INVERT_SAME_RECT:// OK, not tested case PICT.OP_FILL_SAME_RECT:// OK, not tested // Draw switch (opCode) { case PICT.OP_FRAME_RECT: case PICT.OP_FRAME_SAME_RECT: context.frameRect(lastRectangle); break; case PICT.OP_PAINT_RECT: case PICT.OP_PAINT_SAME_RECT: context.paintRect(lastRectangle); break; case PICT.OP_ERASE_RECT: case PICT.OP_ERASE_SAME_RECT: context.eraseRect(lastRectangle); break; case PICT.OP_INVERT_RECT: case PICT.OP_INVERT_SAME_RECT: context.invertRect(lastRectangle); break; case PICT.OP_FILL_RECT: case PICT.OP_FILL_SAME_RECT: context.fillRect(lastRectangle, fill); break; } // Do verbose mode output if (DEBUG) { switch (opCode) { case PICT.OP_FRAME_RECT: System.out.println("frameRect: " + lastRectangle); break; case PICT.OP_PAINT_RECT: System.out.println("paintRect: " + lastRectangle); break; case PICT.OP_ERASE_RECT: System.out.println("eraseRect: " + lastRectangle); break; case PICT.OP_INVERT_RECT: System.out.println("invertRect: " + lastRectangle); break; case PICT.OP_FILL_RECT: System.out.println("fillRect: " + lastRectangle); break; case PICT.OP_FRAME_SAME_RECT: System.out.println("frameSameRect: " + lastRectangle); break; case PICT.OP_PAINT_SAME_RECT: System.out.println("paintSameRect: " + lastRectangle); break; case PICT.OP_ERASE_SAME_RECT: System.out.println("eraseSameRect: " + lastRectangle); break; case PICT.OP_INVERT_SAME_RECT: System.out.println("invertSameRect: " + lastRectangle); break; case PICT.OP_FILL_SAME_RECT: System.out.println("fillSameRect: " + lastRectangle); break; } } // Rect treatments finished break; case 0x003d: case 0x003e: case 0x003f: if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; //-------------------------------------------------------------------------------- // Round Rect treatments //-------------------------------------------------------------------------------- case PICT.OP_FRAME_R_RECT:// OK case PICT.OP_PAINT_R_RECT:// OK, not tested case PICT.OP_ERASE_R_RECT:// OK, not tested case PICT.OP_INVERT_R_RECT:// OK, not tested case PICT.OP_FILL_R_RECT:// OK, not tested // Get the frame rectangle readRectangle(pStream, lastRectangle); case PICT.OP_FRAME_SAME_R_RECT:// OK, not tested case PICT.OP_PAINT_SAME_R_RECT:// OK, not tested case PICT.OP_ERASE_SAME_R_RECT:// OK, not tested case PICT.OP_INVERT_SAME_R_RECT:// OK, not tested case PICT.OP_FILL_SAME_R_RECT:// OK, not tested // Draw switch (opCode) { case PICT.OP_FRAME_R_RECT: case PICT.OP_FRAME_SAME_R_RECT: context.frameRoundRect(lastRectangle, ovSize.x, ovSize.y); break; case PICT.OP_PAINT_R_RECT: case PICT.OP_PAINT_SAME_R_RECT: context.paintRoundRect(lastRectangle, ovSize.x, ovSize.y); break; case PICT.OP_ERASE_R_RECT: case PICT.OP_ERASE_SAME_R_RECT: context.eraseRoundRect(lastRectangle, ovSize.x, ovSize.y); break; case PICT.OP_INVERT_R_RECT: case PICT.OP_INVERT_SAME_R_RECT: context.invertRoundRect(lastRectangle, ovSize.x, ovSize.y); break; case PICT.OP_FILL_R_RECT: case PICT.OP_FILL_SAME_R_RECT: context.fillRoundRect(lastRectangle, ovSize.x, ovSize.y, fill); break; } // Do verbose mode output if (DEBUG) { switch (opCode) { case PICT.OP_FRAME_R_RECT: System.out.println("frameRRect: " + lastRectangle); break; case PICT.OP_PAINT_R_RECT: System.out.println("paintRRect: " + lastRectangle); break; case PICT.OP_ERASE_R_RECT: System.out.println("eraseRRect: " + lastRectangle); break; case PICT.OP_INVERT_R_RECT: System.out.println("invertRRect: " + lastRectangle); break; case PICT.OP_FILL_R_RECT: System.out.println("fillRRect: " + lastRectangle); break; case PICT.OP_FRAME_SAME_R_RECT: System.out.println("frameSameRRect: " + lastRectangle); break; case PICT.OP_PAINT_SAME_R_RECT: System.out.println("paintSameRRect: " + lastRectangle); break; case PICT.OP_ERASE_SAME_R_RECT: System.out.println("eraseSameRRect: " + lastRectangle); break; case PICT.OP_INVERT_SAME_R_RECT: System.out.println("invertSameRRect: " + lastRectangle); break; case PICT.OP_FILL_SAME_R_RECT: System.out.println("fillSameRRect: " + lastRectangle); break; } } // RoundRect treatments finished break; //-------------------------------------------------------------------------------- // Oval treatments //-------------------------------------------------------------------------------- case PICT.OP_FRAME_OVAL:// OK case PICT.OP_PAINT_OVAL:// OK, not tested case PICT.OP_ERASE_OVAL:// OK, not tested case PICT.OP_INVERT_OVAL:// OK, not tested case PICT.OP_FILL_OVAL:// OK, not tested // Get the frame rectangle readRectangle(pStream, lastRectangle); case PICT.OP_FRAME_SAME_OVAL:// OK, not tested case PICT.OP_PAINT_SAME_OVAL:// OK, not tested case PICT.OP_ERASE_SAME_OVAL:// OK, not tested case PICT.OP_INVERT_SAME_OVAL:// OK, not tested case PICT.OP_FILL_SAME_OVAL:// OK, not tested // Draw switch (opCode) { case PICT.OP_FRAME_OVAL: case PICT.OP_FRAME_SAME_OVAL: context.frameOval(lastRectangle); break; case PICT.OP_PAINT_OVAL: case PICT.OP_PAINT_SAME_OVAL: context.paintOval(lastRectangle); break; case PICT.OP_ERASE_OVAL: case PICT.OP_ERASE_SAME_OVAL: context.eraseOval(lastRectangle); break; case PICT.OP_INVERT_OVAL: case PICT.OP_INVERT_SAME_OVAL: context.invertOval(lastRectangle); break; case PICT.OP_FILL_OVAL: case PICT.OP_FILL_SAME_OVAL: context.fillOval(lastRectangle, fill); break; } // Do verbose mode output if (DEBUG) { switch (opCode) { case PICT.OP_FRAME_OVAL: System.out.println("frameOval: " + lastRectangle); break; case PICT.OP_PAINT_OVAL: System.out.println("paintOval: " + lastRectangle); break; case PICT.OP_ERASE_OVAL: System.out.println("eraseOval: " + lastRectangle); break; case PICT.OP_INVERT_OVAL: System.out.println("invertOval: " + lastRectangle); break; case PICT.OP_FILL_OVAL: System.out.println("fillOval: " + lastRectangle); break; case PICT.OP_FRAME_SAME_OVAL: System.out.println("frameSameOval: " + lastRectangle); break; case PICT.OP_PAINT_SAME_OVAL: System.out.println("paintSameOval: " + lastRectangle); break; case PICT.OP_ERASE_SAME_OVAL: System.out.println("eraseSameOval: " + lastRectangle); break; case PICT.OP_INVERT_SAME_OVAL: System.out.println("invertSameOval: " + lastRectangle); break; case PICT.OP_FILL_SAME_OVAL: System.out.println("fillSameOval: " + lastRectangle); break; } } // Oval treatments finished break; case 0x35: case 0x36: case 0x37: case 0x45: case 0x46: case 0x47: case 0x55: case 0x56: case 0x57: pStream.readFully(new byte[8], 0, 8); if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; //-------------------------------------------------------------------------------- // Arc treatments //-------------------------------------------------------------------------------- case PICT.OP_FRAME_ARC:// OK, not tested case PICT.OP_PAINT_ARC:// OK, not tested case PICT.OP_ERASE_ARC:// OK, not tested case PICT.OP_INVERT_ARC:// OK, not tested case PICT.OP_FILL_ARC:// OK, not tested // Get the frame rectangle readRectangle(pStream, lastRectangle); case PICT.OP_FRAME_SAME_ARC:// OK, not tested case PICT.OP_PAINT_SAME_ARC:// OK, not tested case PICT.OP_ERASE_SAME_ARC:// OK, not tested case PICT.OP_INVERT_SAME_ARC:// OK, not tested case PICT.OP_FILL_SAME_ARC:// OK, not tested // NOTE: These are inlcuded even if SAME // Get start and end angles //x = getXPtCoord(pStream.readUnsignedShort()); //y = getYPtCoord(pStream.readUnsignedShort()); x = pStream.readUnsignedShort(); y = pStream.readUnsignedShort(); arcAngles.setLocation(x, y); // Draw switch (opCode) { case PICT.OP_FRAME_ARC: case PICT.OP_FRAME_SAME_ARC: context.frameArc(lastRectangle, arcAngles.x, arcAngles.y); break; case PICT.OP_PAINT_ARC: case PICT.OP_PAINT_SAME_ARC: context.paintArc(lastRectangle, arcAngles.x, arcAngles.y); break; case PICT.OP_ERASE_ARC: case PICT.OP_ERASE_SAME_ARC: context.eraseArc(lastRectangle, arcAngles.x, arcAngles.y); break; case PICT.OP_INVERT_ARC: case PICT.OP_INVERT_SAME_ARC: context.invertArc(lastRectangle, arcAngles.x, arcAngles.y); break; case PICT.OP_FILL_ARC: case PICT.OP_FILL_SAME_ARC: context.fillArc(lastRectangle, arcAngles.x, arcAngles.y, fill); break; } // Do verbose mode output if (DEBUG) { switch (opCode) { case PICT.OP_FRAME_ARC: System.out.println("frameArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_PAINT_ARC: System.out.println("paintArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_ERASE_ARC: System.out.println("eraseArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_INVERT_ARC: System.out.println("invertArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_FILL_ARC: System.out.println("fillArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_FRAME_SAME_ARC: System.out.println("frameSameArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_PAINT_SAME_ARC: System.out.println("paintSameArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_ERASE_SAME_ARC: System.out.println("eraseSameArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_INVERT_SAME_ARC: System.out.println("invertSameArc: " + lastRectangle + ", angles:" + arcAngles); break; case PICT.OP_FILL_SAME_ARC: System.out.println("fillSameArc: " + lastRectangle + ", angles:" + arcAngles); break; } } // Arc treatments finished break; case 0x65: case 0x66: case 0x67: pStream.readFully(new byte[12], 0, 12); if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; case 0x6d: case 0x6e: case 0x6f: pStream.readFully(new byte[4], 0, 4); if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; //-------------------------------------------------------------------------------- // Polygon treatments //-------------------------------------------------------------------------------- case PICT.OP_FRAME_POLY:// OK case PICT.OP_PAINT_POLY:// OK case PICT.OP_ERASE_POLY:// OK, not tested case PICT.OP_INVERT_POLY:// OK, not tested case PICT.OP_FILL_POLY:// OK, not tested // Read the polygon polygon = readPoly(pStream, bounds); case PICT.OP_FRAME_SAME_POLY:// OK, not tested case PICT.OP_PAINT_SAME_POLY:// OK, not tested case PICT.OP_ERASE_SAME_POLY:// OK, not tested case PICT.OP_INVERT_SAME_POLY:// OK, not tested case PICT.OP_FILL_SAME_POLY:// OK, not tested // Draw switch (opCode) { case PICT.OP_FRAME_POLY: case PICT.OP_FRAME_SAME_POLY: context.framePoly(polygon); break; case PICT.OP_PAINT_POLY: case PICT.OP_PAINT_SAME_POLY: context.paintPoly(polygon); break; case PICT.OP_ERASE_POLY: case PICT.OP_ERASE_SAME_POLY: context.erasePoly(polygon); break; case PICT.OP_INVERT_POLY: case PICT.OP_INVERT_SAME_POLY: context.invertPoly(polygon); break; case PICT.OP_FILL_POLY: case PICT.OP_FILL_SAME_POLY: context.fillPoly(polygon, fill); break; } // Do verbose mode output if (DEBUG) { switch (opCode) { case PICT.OP_FRAME_POLY: verbosePolyCmd("framePoly", bounds, polygon); break; case PICT.OP_PAINT_POLY: verbosePolyCmd("paintPoly", bounds, polygon); break; case PICT.OP_ERASE_POLY: verbosePolyCmd("erasePoly", bounds, polygon); break; case PICT.OP_INVERT_POLY: verbosePolyCmd("invertPoly", bounds, polygon); break; case PICT.OP_FILL_POLY: verbosePolyCmd("fillPoly", bounds, polygon); break; case PICT.OP_FRAME_SAME_POLY: verbosePolyCmd("frameSamePoly", bounds, polygon); break; case PICT.OP_PAINT_SAME_POLY: verbosePolyCmd("paintSamePoly", bounds, polygon); break; case PICT.OP_ERASE_SAME_POLY: verbosePolyCmd("eraseSamePoly", bounds, polygon); break; case PICT.OP_INVERT_SAME_POLY: verbosePolyCmd("invertSamePoly", bounds, polygon); break; case PICT.OP_FILL_SAME_POLY: verbosePolyCmd("fillSamePoly", bounds, polygon); break; } } // Polygon treatments finished break; case 0x7d: case 0x7e: case 0x7f: if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; case 0x75: case 0x76: case 0x77: // Read the polygon polygon = readPoly(pStream, bounds); if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; //-------------------------------------------------------------------------------- // Region treatments //-------------------------------------------------------------------------------- case PICT.OP_FRAME_RGN:// OK, not tested case PICT.OP_PAINT_RGN:// OK, not tested case PICT.OP_ERASE_RGN:// OK, not tested case PICT.OP_INVERT_RGN:// OK, not tested case PICT.OP_FILL_RGN:// OK, not tested // Read the region region = readRegion(pStream, bounds); case PICT.OP_FRAME_SAME_RGN:// OK, not tested case PICT.OP_PAINT_SAME_RGN:// OK, not tested case PICT.OP_ERASE_SAME_RGN:// OK, not tested case PICT.OP_INVERT_SAME_RGN:// OK, not tested case PICT.OP_FILL_SAME_RGN:// OK, not tested // Draw if (region != null && !region.getBounds().isEmpty()) { switch (opCode) { case PICT.OP_FRAME_RGN: case PICT.OP_FRAME_SAME_RGN: context.frameRegion(region); break; case PICT.OP_PAINT_RGN: case PICT.OP_PAINT_SAME_RGN: context.paintRegion(region); break; case PICT.OP_ERASE_RGN: case PICT.OP_ERASE_SAME_RGN: context.eraseRegion(region); break; case PICT.OP_INVERT_RGN: case PICT.OP_INVERT_SAME_RGN: context.invertRegion(region); break; case PICT.OP_FILL_RGN: case PICT.OP_FILL_SAME_RGN: context.fillRegion(region, fill); break; } } // Do verbose mode output if (DEBUG) { switch (opCode) { case PICT.OP_FRAME_RGN: verboseRegionCmd("frameRgn", bounds, region); break; case PICT.OP_PAINT_RGN: verboseRegionCmd("paintRgn", bounds, region); break; case PICT.OP_ERASE_RGN: verboseRegionCmd("eraseRgn", bounds, region); break; case PICT.OP_INVERT_RGN: verboseRegionCmd("invertRgn", bounds, region); break; case PICT.OP_FILL_RGN: verboseRegionCmd("fillRgn", bounds, region); break; case PICT.OP_FRAME_SAME_RGN: verboseRegionCmd("frameSameRgn", bounds, region); break; case PICT.OP_PAINT_SAME_RGN: verboseRegionCmd("paintSameRgn", bounds, region); break; case PICT.OP_ERASE_SAME_RGN: verboseRegionCmd("eraseSameRgn", bounds, region); break; case PICT.OP_INVERT_SAME_RGN: verboseRegionCmd("invertSameRgn", bounds, region); break; case PICT.OP_FILL_SAME_RGN: verboseRegionCmd("fillSameRgn", bounds, region); break; } } // Region treatments finished break; case 0x85: case 0x86: case 0x87: // Read the region region = readRegion(pStream, bounds); if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; case PICT.OP_BITS_RECT: // [4] Four opcodes ($0090, $0091, $0098, $0099) are modifications of version 1 opcodes. // The first word following the opcode is rowBytes. If the high bit of rowBytes is set, // then it is a pixel map containing multiple bits per pixel; if it is not set, it is a // bitmap containing 1 bit per pixel. // In general, the difference between version 2 and version 1 formats is that the pixel // map replaces the bitmap, a color table has been added, and pixData replaces bitData. // [5] For opcodes $0090 (BitsRect) and $0091 (BitsRgn), the data is unpacked. These // opcodes can be used only when rowBytes is less than 8. readOpBits(pStream, false); break; case PICT.OP_BITS_RGN: readOpBits(pStream, true); break; case 0x92: case 0x93: case 0x94: case 0x95: case 0x96: case 0x97: dataLength = pStream.readUnsignedShort(); pStream.readFully(new byte[dataLength], 0, dataLength); if (DEBUG) { System.out.printf("%s: 0x%04x - length: %d%n", PICT.APPLE_USE_RESERVED_FIELD, opCode, dataLength); } break; case PICT.OP_PACK_BITS_RECT: readOpPackBits(pStream, false, pixmapCount++); break; case PICT.OP_PACK_BITS_RGN: readOpPackBits(pStream, true, pixmapCount++); break; case PICT.OP_DIRECT_BITS_RECT: readOpDirectBits(pStream, false, pixmapCount++); break; case PICT.OP_DIRECT_BITS_RGN: readOpDirectBits(pStream, true, pixmapCount++); break; case 0x9C: case 0x9D: case 0x9E: case 0x9F: // TODO: Move to special Apple Reserved handling? dataLength = pStream.readUnsignedShort(); pStream.readFully(new byte[dataLength], 0, dataLength); if (DEBUG) { System.out.printf("%s: 0x%04x%n", PICT.APPLE_USE_RESERVED_FIELD, opCode); } break; case PICT.OP_SHORT_COMMENT:// NOTHING TO DO, JUST JUMP OVER byte[] shortComment = new byte[2]; pStream.readFully(shortComment, 0, 2); if (DEBUG) { System.out.println("Short comment: " + Arrays.toString(shortComment)); } break; case PICT.OP_LONG_COMMENT:// NOTHING TO DO, JUST JUMP OVER /*byte[] longComment =*/ readLongComment(pStream); // TODO: Don't just skip... // https://developer.apple.com/legacy/library/documentation/mac/pdf/Imaging_With_QuickDraw/Appendix_B.pdf // Long comments can be used for PhotoShop IRBs (kind 498) or ICC profiles (224) and other meta data... // if (DEBUG) { // System.out.println("Long comment: " + Arrays.toString(longComment)); // } break; case PICT.OP_END_OF_PICTURE:// OK break; case PICT.OP_COMPRESSED_QUICKTIME: // $8200: CompressedQuickTime Data length (Long), data (private to QuickTime) 4 + data length readCompressedQT(pStream); break; case PICT.OP_UNCOMPRESSED_QUICKTIME:// JUST JUMP OVER // $8201: UncompressedQuickTime Data length (Long), data (private to QuickTime) 4 + data length // TODO: Read this as well, need test data dataLength = pStream.readInt(); if (DEBUG) { System.out.printf("unCompressedQuickTime, length %d%n", dataLength); } pStream.readFully(new byte[dataLength], 0, dataLength); break; default: // See: http://developer.apple.com/DOCUMENTATION/mac/QuickDraw/QuickDraw-461.html if (opCode >= 0x00a2 && opCode <= 0x00af) { dataLength = pStream.readUnsignedShort(); } else if (opCode >= 0x00b0 && opCode <= 0x00cf) { // Zero-length dataLength = 0; } else if (opCode >= 0x00d0 && opCode <= 0x00fe) { dataLength = pStream.readInt(); } else if (opCode >= 0x0100 && opCode <= 0x7fff) { // For opcodes $0100-$7FFF, the amount of data for // opcode $nnXX = 2 times nn bytes. dataLength = (opCode & 0xff00) >> 7; } else if (opCode >= 0x8000 && opCode <= 0x80ff) { // Zero-length dataLength = 0; } else if (opCode >= 0x8100 && opCode <= 0x81ff) { dataLength = pStream.readInt(); } else if (opCode == 0xffff) { dataLength = pStream.readInt(); } else { throw new IIOException(String.format("Found unknown opcode: 0x%04x", opCode)); } if (DEBUG) { System.out.printf("%s: 0x%04x - length: %s%n", PICT.APPLE_USE_RESERVED_FIELD, opCode, dataLength); } if (dataLength != 0) { pStream.readFully(new byte[dataLength], 0, dataLength); } } } while (opCode != PICT.OP_END_OF_PICTURE); } catch (IIOException e) { throw e; } catch (EOFException e) { String pos; try { pos = String.format("position %d", imageInput.getStreamPosition()); } catch (IOException ignore) { pos = "unknown position"; } throw new IIOException(String.format("Error in PICT format: Unexpected end of File at %s", pos), e); } catch (IOException e) { throw new IIOException(String.format("Error in PICT format: %s", e.getMessage()), e); } } /* http://devworld.apple.com/documentation/QuickTime/RM/CompressDecompress/ImageComprMgr/B-Chapter/chapter_1000_section_5.html: Field name Description Data size (in bytes) Opcode Compressed picture data 2 Size Size in bytes of data for this opcode 4 Version Version of this opcode 2 Matrix 3 by 3 fixed transformation matrix 36 MatteSize Size of matte data in bytes 4 MatteRect Rectangle for matte data 8 Mode Transfer mode 2 SrcRect Rectangle for source 8 Accuracy Preferred accuracy 4 MaskSize Size of mask region in bytes 4 */ private void readCompressedQT(final ImageInputStream pStream) throws IOException { int dataLength = pStream.readInt(); long pos = pStream.getStreamPosition(); int version = pStream.readUnsignedShort(); if (DEBUG) { System.out.print("compressedQuickTime"); System.out.println(", size: " + dataLength + ", version: " + version); } // Matrix int[] matrix = new int[9]; for (int i = 0; i < matrix.length; i++) { matrix[i] = pStream.readInt(); } if (DEBUG) { System.out.printf("matrix: %s%n", Arrays.toString(matrix)); } // Matte long matteSize = pStream.readUnsignedInt(); Rectangle matteRect = new Rectangle(); readRectangle(pStream, matteRect); // Transfer mode int mode = pStream.readUnsignedShort(); // Read the source rectangle Rectangle srcRect = new Rectangle(); readRectangle(pStream, srcRect); // ...and more int accuracy = pStream.readInt(); int maskSize = pStream.readInt(); if (DEBUG) { System.out.print("matteSize: " + matteSize); System.out.print(", matteRect: " + matteRect); System.out.print(", mode: " + mode); System.out.print(", srcRect: " + srcRect); System.out.print(", accuracy: " + accuracy); System.out.println(", maskSize: " + maskSize); } BufferedImage image = QuickTime.decompress(pStream); if (image != null) { context.copyBits(image, srcRect, srcRect, mode, null); pStream.seek(pos + dataLength); // Might be word-align mismatch here // Skip "QuickTime™ and a ... decompressor required" text // NOTE: This algorithm is reverse-engineered by looking at various test inputs and not from any spec... int penSizeMagic = pStream.readInt(); if (penSizeMagic == 0x000700ae) { // OP_PN_SIZE + bogus x value..? int skip = pStream.readUnsignedShort(); // bogus y value is the number of bytes to skip pStream.skipBytes(skip); // Following opcode should be a OP_PN_SIZE with real values } else { pStream.seek(pos + dataLength); } } else { pStream.seek(pos + dataLength); } } /* http://devworld.apple.com/documentation/QuickTime/RM/CompressDecompress/ImageComprMgr/B-Chapter/chapter_1000_section_5.html: Field name Description Data size (in bytes) Opcode Uncompressed picture data 2 Size Size in bytes of data for this opcode 4 Version Version of this opcode 2 Matrix 3 by 3 fixed transformation matrix 36 MatteSize Size of matte data in bytes 4 MatteRect Rectangle for matte data 8 */ private void readOpPackBits(final ImageInputStream pStream, final boolean hasRegion, final int pPixmapCount) throws IOException { // Get rowBytes int rowBytesRaw = pStream.readUnsignedShort(); int rowBytes = rowBytesRaw & 0x7FFF; boolean isPixMap = (rowBytesRaw & 0x8000) > 0; if (DEBUG) { System.out.print(hasRegion ? "packBitsRgn" : "packBitsRect"); System.out.print(", rowBytes: " + rowBytes); if (isPixMap) { System.out.print(", it is a PixMap"); } else { System.out.print(", it is a BitMap"); } } // Get bounds rectangle. THIS IS NOT TO BE SCALED BY THE RESOLUTION! Rectangle bounds = new Rectangle(); int y = pStream.readUnsignedShort(); int x = pStream.readUnsignedShort(); bounds.setLocation(x, y); y = pStream.readUnsignedShort(); x = pStream.readUnsignedShort(); bounds.setSize(x - bounds.x, y - bounds.y); if (DEBUG) { System.out.print(", bounds: " + bounds); } ColorModel colorModel; int cmpSize; if (isPixMap) { // Get PixMap record version number int pmVersion = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", pmVersion: " + pmVersion); } // Get packing format int packType = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", packType: " + packType); } // Get size of packed data (not used for v2) int packSize = pStream.readInt(); if (DEBUG) { System.out.println(", packSize: " + packSize); } // Get resolution info double hRes = PICTUtil.readFixedPoint(pStream); double vRes = PICTUtil.readFixedPoint(pStream); if (DEBUG) { System.out.print("hRes: " + hRes + ", vRes: " + vRes); } // Get pixel type int pixelType = pStream.readUnsignedShort(); if (DEBUG) { if (pixelType == 0) { System.out.print(", indexed pixels"); } else { System.out.print(", RGBDirect"); } } // Get pixel size int pixelSize = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", pixelSize:" + pixelSize); } // Get pixel component count int cmpCount = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", cmpCount:" + cmpCount); } // Get pixel component size cmpSize = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", cmpSize:" + cmpSize); } // planeBytes (ignored) int planeBytes = pStream.readInt(); if (DEBUG) { System.out.print(", planeBytes:" + planeBytes); } // Handle to ColorTable record int clutId = pStream.readInt(); if (DEBUG) { System.out.print(", clutId:" + clutId); } // Reserved pStream.readInt(); // TODO: Seems to be packType 0 all the time? // packType = 0 means default.... if (packType != 0) { throw new IIOException("Unknown pack type: " + packType); } if (pixelType != 0) { throw new IIOException("Unsupported pixel type: " + pixelType); } // Color table colorModel = PICTUtil.readColorTable(pStream, pixelSize); } else { // Old style BitMap record cmpSize = 1; colorModel = QuickDraw.MONOCHROME; } // Get source rectangle. We DO NOT scale the coordinates by the // resolution info, since we are in pixmap coordinates here // TODO: readReactangleNonScaled() Rectangle srcRect = new Rectangle(); y = pStream.readUnsignedShort(); x = pStream.readUnsignedShort(); srcRect.setLocation(x, y); y = pStream.readUnsignedShort(); x = pStream.readUnsignedShort(); srcRect.setSize(x - srcRect.x, y - srcRect.y); if (DEBUG) { System.out.print(", srcRect:" + srcRect); } // TODO: FixMe... // Get destination rectangle. We DO scale the coordinates according to // the image resolution, since we are working in display coordinates Rectangle dstRect = new Rectangle(); readRectangle(pStream, dstRect); if (DEBUG) { System.out.print(", dstRect:" + dstRect); } // Get transfer mode int transferMode = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", mode: " + transferMode); } Rectangle regionBounds = new Rectangle(); Area region = hasRegion ? readRegion(pStream, regionBounds) : null; if (DEBUG) { if (hasRegion) { verboseRegionCmd(", region", regionBounds, region); } else { System.out.println(); } } // Set up pixel buffer for the RGB values byte[] pixArray = new byte[srcRect.height * rowBytes]; int pixBufOffset = 0; // Read in the RGB arrays for (int scanline = 0; scanline < srcRect.height; scanline++) { // Read in the scanline if (rowBytes >= 8) { // Get byteCount of the scanline int packedBytesCount = rowBytes > 250 ? pStream.readUnsignedShort() : pStream.readUnsignedByte(); // Unpack them all Decoder decoder = new PackBitsDecoder(); DataInput unPackBits = new DataInputStream(new DecoderStream(IIOUtil.createStreamAdapter(pStream, packedBytesCount), decoder)); unPackBits.readFully(pixArray, pixBufOffset, rowBytes); } else { // Uncompressed imageInput.readFully(pixArray, pixBufOffset, rowBytes); } // Increment pixel buffer offset pixBufOffset += rowBytes; //////////////////////////////////////////////////// // TODO: This works for single image PICTs only... // However, this is the most common case. Ok for now processImageProgress(scanline * 100 / (float) srcRect.height); if (abortRequested()) { processReadAborted(); // Skip rest of image data for (int skip = scanline + 1; skip < srcRect.height; skip++) { // Get byteCount of the scanline int packedBytesCount; if (rowBytes < 8) { packedBytesCount = rowBytes; } else if (rowBytes > 250) { packedBytesCount = pStream.readUnsignedShort(); } else { packedBytesCount = pStream.readUnsignedByte(); } pStream.readFully(new byte[packedBytesCount], 0, packedBytesCount); if (DEBUG) { System.out.print("Skip " + skip + ", byteCount: " + packedBytesCount); } } break; } //////////////////////////////////////////////////// } // We add all new images to it. If we are just replaying, then // "pPixmapCount" will never be greater than the size of the vector if (images.size() <= pPixmapCount) { // Create BufferedImage and add buffer it for multiple reads DataBuffer db = new DataBufferByte(pixArray, pixArray.length); WritableRaster raster = Raster.createPackedRaster(db, (rowBytes * 8) / cmpSize, srcRect.height, cmpSize, null); BufferedImage img = new BufferedImage(colorModel, raster, colorModel.isAlphaPremultiplied(), null); images.add(img); } // Draw the image BufferedImage img = images.get(pPixmapCount); if (img != null) { srcRect.setLocation(0, 0); // Raster always start at 0,0 context.copyBits(img, srcRect, dstRect, transferMode, region); } // Line break at the end if (DEBUG) { System.out.println(); } } /** * Reads the data following a {@code directBitsRect} opcode. * * @param pStream the stream to read from * @param pPixmapCount the index of the bitmap in the PICT file, used for * cahcing. * * @throws IIOException if the data can not be read. * @throws IOException if an I/O error occurs while reading the image. */ private void readOpDirectBits(final ImageInputStream pStream, final boolean hasRegion, final int pPixmapCount) throws IOException { // Skip PixMap pointer (always 0x000000FF); pStream.readInt(); // Get rowBytes int rowBytesRaw = pStream.readUnsignedShort(); int rowBytes = rowBytesRaw & 0x7FFF; boolean isPixMap = (rowBytesRaw & 0x8000) > 0; if (DEBUG) { System.out.print(hasRegion ? "directBitsRgn" : "directBitsRect"); System.out.print(", rowBytes: " + rowBytes); if (isPixMap) { System.out.print(", it is a PixMap"); } else { System.out.print(", it is a BitMap"); } } // Get bounds rectangle. THIS IS NOT TO BE SCALED BY THE RESOLUTION! Rectangle bounds = new Rectangle(); int y = pStream.readUnsignedShort(); int x = pStream.readUnsignedShort(); bounds.setLocation(x, y); y = pStream.readUnsignedShort(); x = pStream.readUnsignedShort(); bounds.setSize(x - bounds.x, y - bounds.y); if (DEBUG) { System.out.print(", bounds: " + bounds); } ColorModel colorModel = null; int packType; int cmpCount; if (isPixMap) { // Get PixMap record version number int pmVersion = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", pmVersion: " + pmVersion); } // Get packing format packType = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", packType: " + packType); } // Get size of packed data (not used for v2) int packSize = pStream.readInt(); if (DEBUG) { System.out.println(", packSize: " + packSize); } // Get resolution info double hRes = PICTUtil.readFixedPoint(pStream); double vRes = PICTUtil.readFixedPoint(pStream); if (DEBUG) { System.out.print("hRes: " + hRes + ", vRes: " + vRes); } // Get pixel type int pixelType = pStream.readUnsignedShort(); if (DEBUG) { if (pixelType == 0) { System.out.print(", indexed pixels"); } else { System.out.print(", RGBDirect"); } } // Get pixel size int pixelSize = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", pixelSize:" + pixelSize); } // Get pixel component count cmpCount = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", cmpCount:" + cmpCount); } // Get pixel component size int cmpSize = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", cmpSize:" + cmpSize); } // planeBytes (ignored) pStream.readInt(); // Handle to ColorTable record, there should be none for direct // bits so this should be 0, just skip pStream.readInt(); // Reserved pStream.readInt(); } else { // Old style BitMap record packType = 1; cmpCount = 1; colorModel = QuickDraw.MONOCHROME; } // Get source rectangle. We DO NOT scale the coordinates by the // resolution info, since we are in pixmap coordinates here Rectangle srcRect = new Rectangle(); y = pStream.readUnsignedShort(); x = pStream.readUnsignedShort(); srcRect.setLocation(x, y); y = pStream.readUnsignedShort(); x = pStream.readUnsignedShort(); srcRect.setSize(x - srcRect.x, y - srcRect.y); if (DEBUG) { System.out.print(", srcRect:" + srcRect); } // Get destination rectangle. We DO scale the coordinates according to // the image resolution, since we are working in display coordinates Rectangle dstRect = new Rectangle(); readRectangle(pStream, dstRect); if (DEBUG) { System.out.print(", dstRect:" + dstRect); } // Get transfer mode int transferMode = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", mode: " + transferMode); } Area region = hasRegion ? readRegion(pStream, new Rectangle()) : null; if (DEBUG) { System.out.println(hasRegion ? ", region: " + region : ""); } // Set up pixel buffer for the RGB values // Read in the RGB arrays byte[] dstBytes; if (packType == 1 || packType == 2 || packType == 3) { dstBytes = new byte[rowBytes]; } else if (packType == 4) { dstBytes = new byte[cmpCount * rowBytes / 4]; } else { throw new IIOException("Unknown pack type: " + packType); } int[] pixArray = null; short[] shortArray = null; if (packType == 3) { shortArray = new short[srcRect.height * (rowBytes + 1) / 2]; } else { pixArray = new int[srcRect.height * (rowBytes + 3) / 4]; } int pixBufOffset = 0; int packedBytesCount; for (int scanline = 0; scanline < srcRect.height; scanline++) { // Read in the scanline if (packType > 2) { // Get byteCount of the scanline if (rowBytes > 250) { packedBytesCount = pStream.readUnsignedShort(); } else { packedBytesCount = pStream.readUnsignedByte(); } if (DEBUG) { System.out.print("Line " + scanline + ", byteCount: " + packedBytesCount); System.out.print(" dstBytes: " + dstBytes.length); System.out.println(); } // Unpack them all Decoder decoder = packType == 3 ? new PackBitsDecoder(2, false) : new PackBitsDecoder(); DataInput unPackBits = new DataInputStream(new DecoderStream(IIOUtil.createStreamAdapter(pStream, packedBytesCount), decoder, dstBytes.length)); unPackBits.readFully(dstBytes); } else { imageInput.readFully(dstBytes); } if (packType == 3) { // TYPE_USHORT_555_RGB for 16 bit for (int i = 0; i < srcRect.width; i++) { shortArray[pixBufOffset + i] = (short) (((0xff & dstBytes[2 * i]) << 8) | (0xff & dstBytes[2 * i + 1])); } } else { if (cmpCount == 3) { // RGB for (int i = 0; i < srcRect.width; i++) { // Set alpha values to all opaque pixArray[pixBufOffset + i] = 0xFF000000 // Get red values | (dstBytes[/*0* bounds.width*/i] & 0xFF) << 16 // Get green values | (dstBytes[/**/bounds.width + i] & 0xFF) << 8 // Get blue values | (dstBytes[2 * bounds.width + i] & 0xFF); } } else if (cmpCount == 4) { // ARGB for (int i = 0; i < srcRect.width; i++) { // Get alpha values pixArray[pixBufOffset + i] = (dstBytes[/*0* bounds.width*/i] & 0xFF) << 24 // Get red values | (dstBytes[/**/bounds.width + i] & 0xFF) << 16 // Get green values | (dstBytes[2 * bounds.width + i] & 0xFF) << 8 // Get blue values | (dstBytes[3 * bounds.width + i] & 0xFF); } } } // Increment pixel buffer offset pixBufOffset += srcRect.width; //////////////////////////////////////////////////// // TODO: This works for single image PICTs only... // However, this is the most common case. Ok for now processImageProgress(scanline * 100 / (float) srcRect.height); if (abortRequested()) { processReadAborted(); // Skip rest of image data for (int skip = scanline + 1; skip < srcRect.height; skip++) { // Get byteCount of the scanline if (rowBytes > 250) { packedBytesCount = pStream.readUnsignedShort(); } else { packedBytesCount = pStream.readUnsignedByte(); } pStream.readFully(new byte[packedBytesCount], 0, packedBytesCount); if (DEBUG) { System.out.println(); System.out.print("Skip " + skip + ", byteCount: " + packedBytesCount); } } break; } //////////////////////////////////////////////////// } // We add all new images to it. If we are just replaying, then // "pPixmapCount" will never be greater than the size of the vector if (images.size() <= pPixmapCount) { // Create BufferedImage and add buffer it for multiple reads ColorModel cm; WritableRaster raster; if (colorModel != null) { cm = colorModel; DataBuffer db = new DataBufferByte(dstBytes, dstBytes.length); raster = Raster.createPackedRaster(db, srcRect.width, srcRect.height, 1, null); } else if (packType == 3) { cm = new DirectColorModel(15, 0x7C00, 0x03E0, 0x001F); // See BufferedImage TYPE_USHORT_555_RGB DataBuffer db = new DataBufferUShort(shortArray, shortArray.length); raster = Raster.createPackedRaster(db, srcRect.width, srcRect.height, srcRect.width, ((DirectColorModel) cm).getMasks(), null); } else { cm = ColorModel.getRGBdefault(); DataBuffer db = new DataBufferInt(pixArray, pixArray.length); raster = Raster.createPackedRaster(db, srcRect.width, srcRect.height, srcRect.width, ((DirectColorModel) cm).getMasks(), null); } BufferedImage img = new BufferedImage(cm, raster, cm.isAlphaPremultiplied(), null); images.add(img); } // Draw the image BufferedImage img = images.get(pPixmapCount); if (img != null) { srcRect.setLocation(0, 0); // Raster always starts at 0,0 context.copyBits(img, srcRect, dstRect, transferMode, region); } // Line break at the end if (DEBUG) { System.out.println(); } } /* pixMap: PixMap; colorTable: ColorTable; srcRect: Rect; {source rectangle} dstRect: Rect; {destination rectangle} mode: Word; {transfer mode (may } { include new modes)} maskRgn: Rgn; {region for masking} pixData: PixData; */ private void readOpBits(ImageInputStream pStream, boolean hasRegion) throws IOException { // Get rowBytes int rowBytesRaw = pStream.readUnsignedShort(); int rowBytes = rowBytesRaw & 0x7FFF; boolean isPixMap = (rowBytesRaw & 0x8000) > 0; if (DEBUG) { System.out.print(hasRegion ? "bitsRgn" : "bitsRect"); System.out.print(", rowBytes: " + rowBytes); if (isPixMap) { System.out.print(", it is a PixMap"); } else { System.out.print(", it is a BitMap"); } } // Get bounds rectangle. THIS IS NOT TO BE SCALED BY THE RESOLUTION! Rectangle bounds = new Rectangle(); int y = pStream.readUnsignedShort(); int x = pStream.readUnsignedShort(); bounds.setLocation(x, y); y = pStream.readUnsignedShort(); x = pStream.readUnsignedShort(); bounds.setSize(x - bounds.x, y - bounds.y); ColorModel colorModel; int cmpSize; if (isPixMap) { // Get PixMap record version number int pmVersion = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", pmVersion: " + pmVersion); } // Get packing format int packType = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", packType: " + packType); } // Get size of packed data (not used for v2) int packSize = pStream.readInt(); if (DEBUG) { System.out.println(", packSize: " + packSize); } // Get resolution info double hRes = PICTUtil.readFixedPoint(pStream); double vRes = PICTUtil.readFixedPoint(pStream); if (DEBUG) { System.out.print("hRes: " + hRes + ", vRes: " + vRes); } // Get pixel type int pixelType = pStream.readUnsignedShort(); if (DEBUG) { if (pixelType == 0) { System.out.print(", indexed pixels"); } else { System.out.print(", RGBDirect"); } } // Get pixel size int pixelSize = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", pixelSize:" + pixelSize); } // Get pixel component count int cmpCount = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", cmpCount:" + cmpCount); } // Get pixel component size cmpSize = pStream.readUnsignedShort(); if (DEBUG) { System.out.print(", cmpSize:" + cmpSize); } // planeBytes (ignored) int planeBytes = pStream.readInt(); if (DEBUG) { System.out.print(", planeBytes:" + planeBytes); } // Handle to ColorTable record int clutId = pStream.readInt(); if (DEBUG) { System.out.print(", clutId:" + clutId); } // Reserved pStream.readInt(); // Color table colorModel = PICTUtil.readColorTable(pStream, pixelSize); } else { // Old style BitMap record colorModel = QuickDraw.MONOCHROME; cmpSize = 1; } Rectangle srcRect = new Rectangle(); readRectangle(pStream, srcRect); Rectangle dstRect = new Rectangle(); readRectangle(pStream, dstRect); // Get transfer mode int mode = pStream.readUnsignedShort(); Area region = hasRegion ? readRegion(pStream, new Rectangle()) : null; if (DEBUG) { System.out.print(", bounds: " + bounds); System.out.print(", srcRect: " + srcRect); System.out.print(", dstRect: " + dstRect); System.out.print(", mode: " + mode); System.out.print(hasRegion ? ", region: " + region : ""); System.out.println(); } byte[] data = new byte[rowBytes * bounds.height]; // Read pixel data for (int i = 0; i < bounds.height; i++) { pStream.readFully(data, i * rowBytes, rowBytes); } DataBuffer db = new DataBufferByte(data, data.length); WritableRaster raster = Raster.createPackedRaster(db, (rowBytes * 8) / cmpSize, srcRect.height, cmpSize, null); BufferedImage image = new BufferedImage(colorModel, raster, colorModel.isAlphaPremultiplied(), null); // Draw pixel data srcRect.setLocation(0, 0); // Raster always start at 0,0 context.copyBits(image, srcRect, dstRect, mode, region); } /** * Reads the rectangle location and size from an 8-byte rectangle stream. * * @param pStream the stream to read from * @param pDestRect the rectangle to read into * * @throws NullPointerException if {@code pDestRect} is {@code null} * @throws IOException if an I/O error occurs while reading the image. */ private void readRectangle(DataInput pStream, Rectangle pDestRect) throws IOException { int y = pStream.readUnsignedShort(); int x = pStream.readUnsignedShort(); int h = pStream.readUnsignedShort(); int w = pStream.readUnsignedShort(); pDestRect.setLocation(getXPtCoord(x), getYPtCoord(y)); pDestRect.setSize(getXPtCoord(w - x), getYPtCoord(h - y)); } /** * Read in a region. The input stream should be positioned at the first byte * of the region. {@code pBounds} is a rectangle that will be set to the * region bounds. * The point array may therefore be empty if the region is just a rectangle. * * @param pStream the stream to read from * @param pBounds the bounds rectangle to read into * * @return the area containing the region, or an empty polygon if the * region is a rectangle. * * @throws IOException if an I/O error occurs while reading the image. */ private Area readRegion(DataInput pStream, Rectangle pBounds) throws IOException { // Get region data size int size = pStream.readUnsignedShort(); // Get region bounds readRectangle(pStream, pBounds); int count = (size - 10) / 2; if (count == 0) { // Minimal region, just the bounds return new Area(pBounds); } else { // Normal region, parse inversion points and build area Area area = new Area(); boolean nextIsVertical = true; int x, y = 0; for (int i = 0; i < count; i++) { short point = pStream.readShort(); if (nextIsVertical) { if (point == 0x7fff) { // Done break; } y = point; nextIsVertical = false; } else { if (point == 0x7fff) { nextIsVertical = true; continue; } x = point; area.exclusiveOr(new Area(new Rectangle(x, y, pBounds.x + pBounds.width - x, pBounds.y + pBounds.height - y))); } } if (!pBounds.contains(area.getBounds())) { processWarningOccurred("Bad region, contains point(s) out of bounds " + pBounds + ": " + area); } return area; } } /** * Read in a polygon. The input stream should be positioned at the first byte * of the polygon. * * @param pStream the stream to read from * @param pBounds the bounds rectangle to read into * * @return the polygon * * @throws IOException if an I/O error occurs while reading the image. */ private Polygon readPoly(DataInput pStream, Rectangle pBounds) throws IOException { // Get polygon data size int size = pStream.readUnsignedShort(); // Get poly bounds readRectangle(pStream, pBounds); // Initialize the point array to the right size int points = (size - 10) / 4; // Get the rest of the polygon points Polygon polygon = new Polygon(); for (int i = 0; i < points; i++) { int y = getYPtCoord(pStream.readShort()); int x = getXPtCoord(pStream.readShort()); polygon.addPoint(x, y); } if (!pBounds.contains(polygon.getBounds())) { processWarningOccurred("Bad poly, contains point(s) out of bounds " + pBounds + ": " + polygon); } return polygon; } // TODO: Support color pixel patterns! /* * Read PixPat. Read a PixPat data structure from the stream. Just returns * void for the moment since not used in AWT graphics. NOT IMPLEMENTED YET! */ /* private void readPixPat(DataInput pStream) { byte[] count = new byte[1]; byte[] text_bytes; try { // Comment kind and data byte count pStream.readFully(count, 0, count.length); // Get as many bytes as indicated by byte count int text_byte_count = count[0]; text_bytes = new byte[text_byte_count]; pStream.readFully(text_bytes, 0, text_byte_count); } catch ( IOException e ) { return null; } return new String(text_bytes); } */ /* * Read a long comment from the stream. */ private byte[] readLongComment(final DataInput pStream) throws IOException { // Comment kind and data byte count short kind = pStream.readShort(); int length = pStream.readUnsignedShort(); if (DEBUG) { System.out.println("Long comment: " + kind + ", " + length + " bytes"); } // Get as many bytes as indicated by byte count byte[] bytes = new byte[length]; pStream.readFully(bytes, 0, length); return bytes; } /* * Return the X coordinate value in display coordinates for the given * coordinate value. This means multiplying it with the screen resolution/ * image resolution ratio. */ private int getXPtCoord(int pPoint) { return (int) Math.round(pPoint / screenImageXRatio); } /* * Return the Y coordinate value in display coordinates for the given * coordinate value. This means multiplying it with the screen resolution/ * image resolution ratio. */ private int getYPtCoord(int pPoint) { return (int) Math.round(pPoint / screenImageYRatio); } /* * Write out polygon command, bounds and points. */ private void verbosePolyCmd(String pCmd, Rectangle pBounds, Polygon pPolygon) { System.out.println(pCmd + ": " + pBounds); System.out.print("Polygon points: "); if (pPolygon != null && pPolygon.npoints > 0) { System.out.print("(" + pPolygon.xpoints[0] + "," + pPolygon.ypoints[0] + ")"); } for (int i = 1; pPolygon != null && i < pPolygon.npoints; i++) { System.out.print(", (" + pPolygon.xpoints[i] + "," + pPolygon.ypoints[i] + ")"); } System.out.println(); } /* * Write out region command, bounds and points. */ private void verboseRegionCmd(String pCmd, Rectangle pBounds, Area pRegion) { System.out.println(pCmd + ": " + pBounds); // System.out.print("Region points: "); // if (pPolygon != null && pPolygon.npoints > 0) { // System.out.print("(" + pPolygon.xpoints[0] + "," + pPolygon.ypoints[0] + ")"); // } // for (int i = 1; pPolygon != null && i < pPolygon.npoints; i++) { // System.out.print(", (" + pPolygon.xpoints[i] + "," + pPolygon.ypoints[i] + ")"); // } System.out.println(); } @Override public BufferedImage read(final int pIndex, final ImageReadParam pParam) throws IOException { checkBounds(pIndex); processImageStarted(pIndex); // TODO: Param handling // TODO: Real subsampling for bit/pixmap/QT stills final int subX, subY; if (pParam != null) { subX = pParam.getSourceXSubsampling(); subY = pParam.getSourceYSubsampling(); } else { subX = 1; subY = 1; } Rectangle frame = getPICTFrame(); BufferedImage image = getDestination(pParam, getImageTypes(pIndex), getXPtCoord(frame.width), getYPtCoord(frame.height)); Graphics2D g = image.createGraphics(); try { // Might need to clear background g.setComposite(AlphaComposite.Src); g.setColor(new Color(0x00ffffff, true)); // Transparent white g.fillRect(0, 0, image.getWidth(), image.getHeight()); AffineTransform instance = new AffineTransform(); if (pParam != null && pParam.getSourceRegion() != null) { Rectangle rectangle = pParam.getSourceRegion(); instance.translate(-rectangle.x, -rectangle.y); } instance.scale(screenImageXRatio / subX, screenImageYRatio / subY); instance.translate(-frame.x, -frame.y); g.setTransform(instance); drawOnto(g); } finally { g.dispose(); } processImageComplete(); return image; } public int getWidth(int pIndex) throws IOException { checkBounds(pIndex); return getXPtCoord(getPICTFrame().width); } public int getHeight(int pIndex) throws IOException { checkBounds(pIndex); return getYPtCoord(getPICTFrame().height); } public Iterator getImageTypes(int imageIndex) throws IOException { checkBounds(imageIndex); // TODO: The images look slightly different in Preview.. Could indicate the color space is wrong... return Collections.singletonList( ImageTypeSpecifiers.createPacked( ColorSpace.getInstance(ColorSpace.CS_sRGB), 0xff0000, 0xff00, 0xff, 0xff000000, DataBuffer.TYPE_INT, false ) ).iterator(); } @Override public IIOMetadata getImageMetadata(final int imageIndex) throws IOException { ImageTypeSpecifier rawType = getRawImageType(imageIndex); getPICTFrame(); // TODO: Would probably be better to use readPictHeader here, but it isn't cached return new PICTMetadata(rawType, version, screenImageXRatio, screenImageYRatio); } protected static void showIt(final BufferedImage pImage, final String pTitle) { ImageReaderBase.showIt(pImage, pTitle); } public static void main(final String[] pArgs) { ImageReader reader = new PICTImageReader(new PICTImageReaderSpi()); for (String arg : pArgs) { File file = new File(arg); try { ImageInputStream input = ImageIO.createImageInputStream(file); String title = file.getName(); System.out.println("canRead: " + reader.getOriginatingProvider().canDecodeInput(input)); reader.setInput(input); // BufferedImage image = reader.getImageTypes(0).next().createBufferedImage(reader.getWidth(0), reader.getHeight(0)); // ImageReadParam param = reader.getDefaultReadParam(); // param.setDestination(image); long start = System.currentTimeMillis(); BufferedImage image = reader.read(0); System.out.println("time: " + (System.currentTimeMillis() - start)); showIt(image, title); System.out.println("image = " + image); } catch (IOException e) { System.err.println("Could not read " + file.getAbsolutePath() + ": " + e); } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy