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

com.twelvemonkeys.imageio.plugins.icns.ICNSImageReader Maven / Gradle / Ivy

There is a newer version: 3.12.0
Show newest version
/*
 * Copyright (c) 2011, 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.
 */

package com.twelvemonkeys.imageio.plugins.icns;

import com.twelvemonkeys.imageio.ImageReaderBase;
import com.twelvemonkeys.imageio.util.IIOUtil;
import com.twelvemonkeys.imageio.util.ImageTypeSpecifiers;

import javax.imageio.*;
import javax.imageio.spi.ImageReaderSpi;
import javax.imageio.stream.ImageInputStream;
import java.awt.*;
import java.awt.color.ColorSpace;
import java.awt.image.*;
import java.io.DataInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;

/**
 * ImageReader for Apple Icon Image (ICNS) format.
 *
 * @author Harald Kuhr
 * @author last modified by $Author: haraldk$
 * @version $Id: ICNSImageReader.java,v 1.0 25.10.11 18:42 haraldk Exp$
 *
 * @see Macintosh Icons
 * @see Apple Icon Image format (Wikipedia)
 */
public final class ICNSImageReader extends ImageReaderBase {
    // TODO: Support ToC resource for faster parsing/faster determine number of icons?
    // TODO: Subsampled reading for completeness, even if never used?
    private List icons = new ArrayList();
    private List masks = new ArrayList();
    private IconResource lastResourceRead;

    private int length;

    public ICNSImageReader() {
        this(new ICNSImageReaderSpi());
    }

    ICNSImageReader(final ImageReaderSpi provider) {
        super(provider);
    }

    @Override
    protected void resetMembers() {
        length = 0;

        lastResourceRead = null;
        icons.clear();
        masks.clear();
    }

    @Override
    public int getWidth(int imageIndex) throws IOException {
        return readIconResource(imageIndex).size().width;
    }

    @Override
    public int getHeight(int imageIndex) throws IOException {
        return readIconResource(imageIndex).size().height;
    }

    @Override
    public ImageTypeSpecifier getRawImageType(int imageIndex) throws IOException {
        IconResource resource = readIconResource(imageIndex);

        switch (resource.depth()) {
            case 1:
                return ImageTypeSpecifiers.createFromIndexColorModel(ICNS1BitColorModel.INSTANCE);
            case 4:
                return ImageTypeSpecifiers.createFromIndexColorModel(ICNS4BitColorModel.INSTANCE);
            case 8:
                return ImageTypeSpecifiers.createFromIndexColorModel(ICNS8BitColorModel.INSTANCE);
            case 32:
                if (resource.isCompressed()) {
                    return ImageTypeSpecifiers.createBanded(
                            ColorSpace.getInstance(ColorSpace.CS_sRGB),
                            new int[]{0, 1, 2, 3}, createBandOffsets(resource.size().width * resource.size().height),
                            DataBuffer.TYPE_BYTE, true, false
                    );
                }
                else {
                    return ImageTypeSpecifiers.createInterleaved(
                            ColorSpace.getInstance(ColorSpace.CS_sRGB),
                            new int[]{1, 2, 3, 0},
                            DataBuffer.TYPE_BYTE, true, false
                    );
                }
            default:
                throw new IllegalStateException(String.format("Unknown bit depth: %d", resource.depth()));
        }
    }

    private static int[] createBandOffsets(int bandLen) {
        return new int[]{0, bandLen, 2 * bandLen, 3 * bandLen};
    }

    @Override
    public Iterator getImageTypes(int imageIndex) throws IOException {
        ImageTypeSpecifier rawType = getRawImageType(imageIndex);
        IconResource resource = readIconResource(imageIndex);

        List specifiers = new ArrayList();

        switch (resource.depth()) {
            case 1:
            case 4:
            case 8:
                // Fall through & convert during read
            case 32:
                specifiers.add(ImageTypeSpecifiers.createPacked(ColorSpace.getInstance(ColorSpace.CS_sRGB), 0xff0000, 0x00ff00, 0x0000ff, 0xff000000, DataBuffer.TYPE_INT, false));
                specifiers.add(ImageTypeSpecifiers.createInterleaved(ColorSpace.getInstance(ColorSpace.CS_sRGB), new int[]{3, 2, 1, 0}, DataBuffer.TYPE_BYTE, true, false));
                break;
            default:
                throw new IllegalStateException(String.format("Unknown bit depth: %d", resource.depth()));
        }

        specifiers.add(rawType);

        return specifiers.iterator();
    }

    @Override
    public int getNumImages(boolean allowSearch) throws IOException {
        assertInput();

        if (!allowSearch) {
            // Return icons.size if we know we have read all?
            // TODO: If the first resource is a TOC_ resource, we don't need to perform a search.
            return -1;
        }

        int num = icons.size();
        while (true) {
            try {
                readIconResource(num++);
            }
            catch (IndexOutOfBoundsException expected) {
                break;
            }
        }

        return icons.size();
    }

    @Override
    public BufferedImage read(int imageIndex, ImageReadParam param) throws IOException {
        IconResource resource = readIconResource(imageIndex);

        imageInput.seek(resource.start + ICNS.RESOURCE_HEADER_SIZE);

        // Special handling of PNG/JPEG 2000 icons
        if (resource.isForeignFormat()) {
            return readForeignFormat(imageIndex, param, resource);
        }

        return readICNSFormat(imageIndex, param, resource);
    }

    private BufferedImage readICNSFormat(final int imageIndex, final ImageReadParam param, final IconResource resource) throws IOException {
        Dimension size = resource.size();

        int width = size.width;
        int height = size.height;

        BufferedImage image = getDestination(param, getImageTypes(imageIndex), width, height);
        ImageTypeSpecifier rawType = getRawImageType(imageIndex);

        if (rawType.getColorModel() instanceof IndexColorModel && rawType.getBufferedImageType() != image.getType()) {
            checkReadParamBandSettings(param, 4, image.getSampleModel().getNumBands());
        }
        else {
            checkReadParamBandSettings(param, rawType.getNumBands(), image.getSampleModel().getNumBands());
        }

        final Rectangle source = new Rectangle();
        final Rectangle dest = new Rectangle();
        computeRegions(param, width, height, image, source, dest);

        processImageStarted(imageIndex);

        // Read image data
        byte[] data;
        if (resource.isCompressed()) {
            // Only 32 bit icons may be compressed
            data = new byte[width * height * resource.depth() / 8];

            int packedSize = resource.length - ICNS.RESOURCE_HEADER_SIZE;

            if (width >= 128 && height >= 128) {
                // http://www.macdisk.com/maciconen.php:
                // "In some icon sizes, there is a 32bit integer at the beginning of the run, whose role remains unknown."
                imageInput.skipBytes(4); // Seems to be 4 byte 0-pad
                packedSize -= 4;
            }

            InputStream input = IIOUtil.createStreamAdapter(imageInput, packedSize);

            try {
                ICNSUtil.decompress(new DataInputStream(input), data, 0, (data.length * 24) / 32); // 24 bit data
            }
            finally {
                input.close();
            }
        }
        else {
            data = new byte[resource.length - ICNS.RESOURCE_HEADER_SIZE];
            imageInput.readFully(data);
        }

        if (resource.depth() == 1) {
            // Binary
            DataBufferByte buffer = new DataBufferByte(data, data.length / 2, 0);
            WritableRaster raster = Raster.createPackedRaster(buffer, width, height, resource.depth(), null);

            if (image.getType() == rawType.getBufferedImageType() && ((IndexColorModel) image.getColorModel()).getMapSize() == 2) {
                // Preserve raw data as read (binary), discard mask
                image.setData(raster);
            }
            else {
                // Convert to 32 bit ARGB
                DataBufferByte maskBuffer = new DataBufferByte(data, data.length / 2, data.length / 2);
                WritableRaster mask = Raster.createPackedRaster(maskBuffer, width, height, resource.depth(), null);

                Graphics2D graphics = image.createGraphics();

                try {
                    // Apply image data
                    BufferedImage temp = new BufferedImage(rawType.getColorModel(), raster, false, null);
                    graphics.drawImage(temp, 0, 0, null);

                    // Apply mask
                    temp = new BufferedImage(ICNSBitMaskColorModel.INSTANCE, mask, false, null);
                    temp.setData(mask);
                    graphics.setComposite(AlphaComposite.DstIn);
                    graphics.drawImage(temp, 0, 0, null);
                }
                finally {
                    graphics.dispose();
                }
            }
        }
        else if (resource.depth() <= 8) {
            // Indexed
            DataBufferByte buffer = new DataBufferByte(data, data.length);
            WritableRaster raster = Raster.createPackedRaster(buffer, width, height, resource.depth(), null);
            
            if (image.getType() == rawType.getBufferedImageType()) {
                // Preserve raw data as read (indexed), discard mask
                image.setData(raster);
            }
            else {
                // Convert to 32 bit ARGB
                Graphics2D graphics = image.createGraphics();

                try {
                    BufferedImage temp = new BufferedImage(rawType.getColorModel(), raster, false, null);
                    graphics.drawImage(temp, 0, 0, null);
                }
                finally {
                    graphics.dispose();
                }

                processImageProgress(50f);

                // Read mask and apply
                IconResource maskResource = findMaskResource(resource);

                if (maskResource != null) {
                    Raster mask = readMask(maskResource);
                    image.getAlphaRaster().setRect(mask);
                }
            }
        }
        else {
            // 32 bit ARGB (true color)
            int bandLen = data.length / 4;

            DataBufferByte buffer = new DataBufferByte(data, data.length);

            WritableRaster raster;

            if (resource.isCompressed()) {
                raster = Raster.createBandedRaster(buffer, width, height, width, new int[]{0, 0, 0, 0}, createBandOffsets(bandLen), null);
            }
            else {
                // NOTE: Uncompressed 32bit is interleaved RGBA, not banded...
                raster = Raster.createInterleavedRaster(buffer, width, height, width * 4, 4, new int[]{1, 2, 3, 0}, null);
            }

            image.setData(raster);

            processImageProgress(75f);

            // Read mask and apply
            IconResource maskResource = findMaskResource(resource);

            if (maskResource != null) {
                Raster mask = readMask(maskResource);
                image.getAlphaRaster().setRect(mask);
            }
            else {
                // TODO: This is simply stupid. Rewrite to use no alpha instead?
                byte[] solid = new byte[width * height];
                Arrays.fill(solid, (byte) -1);
                WritableRaster mask = Raster.createBandedRaster(new DataBufferByte(solid, solid.length), width, height, width, new int[]{0}, new int[]{0}, null);
                image.getAlphaRaster().setRect(mask);
            }
        }

        // For now: Make listener tests happy
        // TODO: Implement more sophisticated reading
        processImageProgress(100f);

        if (abortRequested()) {
            processReadAborted();
        }
        else {
            processImageComplete();
        }

        return image;
    }

    private Raster readMask(final IconResource resource) throws IOException {
        Dimension size = resource.size();

        int width = size.width;
        int height = size.height;

        byte[] mask = new byte[width * height];
        imageInput.seek(resource.start + ICNS.RESOURCE_HEADER_SIZE);

        if (resource.isMaskType()) {
            // 8 bit mask
            imageInput.readFully(mask, 0, resource.length - ICNS.RESOURCE_HEADER_SIZE);
        }
        else if (resource.hasMask()) {
            // Embedded 1bit mask
            byte[] maskData = new byte[(resource.length - ICNS.RESOURCE_HEADER_SIZE) / 2];
            imageInput.skipBytes(maskData.length); // Skip the 1 bit image data
            imageInput.readFully(maskData);

            // Unpack 1bit mask to 8 bit
            int bitPos = 0x80;

            for (int i = 0, maskLength = mask.length; i < maskLength; i++) {
                mask[i] = (byte) ((maskData[i / 8] & bitPos) != 0 ? 0xff : 0x00);
                
                if ((bitPos >>= 1) == 0) {
                    bitPos = 0x80;
                }
            }
        }
        else {
            throw new IllegalArgumentException(String.format("Not a mask resource: %s", resource));
        }

        return Raster.createBandedRaster(new DataBufferByte(mask, mask.length), width, height, width, new int[]{0}, new int[]{0}, null);
    }

    private IconResource findMaskResource(final IconResource iconResource) throws IOException {
        // Find 8 bit mask
        try {
            int i = 0;

            while (true) {
                IconResource mask = i < masks.size() ? masks.get(i++) : readNextIconResource();

                if (mask.isMaskType() && mask.size().equals(iconResource.size())) {
                    return mask;
                }
            }
        }
        catch (IndexOutOfBoundsException ignore) {
        }

        // Fall back to mask from 1 bit resource if no 8 bit mask
        for (IconResource resource : icons) {
            if (resource.hasMask() && resource.size().equals(iconResource.size())) {
                return resource;
            }
        }

        return null;
    }

    private BufferedImage readForeignFormat(int imageIndex, final ImageReadParam param, final IconResource resource) throws IOException {
        // TODO: Optimize by caching readers that work?
        ImageInputStream stream = ImageIO.createImageInputStream(IIOUtil.createStreamAdapter(imageInput, resource.length));

        try {
            // Try first using ImageIO
            Iterator readers = ImageIO.getImageReaders(stream);

            while (readers.hasNext()) {
                ImageReader reader = readers.next();
                reader.setInput(stream);

                try {
                    return reader.read(0, param);
                }
                catch (IOException ignore) {
                    if (stream.getFlushedPosition() <= 0) {
                        stream.seek(0);
                    }
                    else {
                        stream.close();
                        stream = ImageIO.createImageInputStream(IIOUtil.createStreamAdapter(imageInput, resource.length));
                    }
                }
                finally {
                    reader.dispose();
                }
            }

            String format = getForeignFormat(stream);

            // OS X quick fix
            if ("JPEG 2000".equals(format) && SipsJP2Reader.isAvailable()) {
                SipsJP2Reader reader = new SipsJP2Reader();
                reader.setInput(stream);
                BufferedImage image = reader.read(0, param);

                if (image != null) {
                    return image;
                }
            }

            // There's no JPEG 2000 reader installed in ImageIO by default (requires JAI ImageIO installed).
            // Return blank icon + issue warning. We know the image dimensions, we just can't read the data.
            processWarningOccurred(String.format(
                    "Cannot read %s format in type '%s' icon (no reader; installed: %s)",
                    format, ICNSUtil.intToStr(resource.type), Arrays.toString(IIOUtil.getNormalizedReaderFormatNames())
            ));

            Dimension size = resource.size();

            return getDestination(param, getImageTypes(imageIndex), size.width, size.height);
        }
        finally {
            stream.close();
        }
    }

    private String getForeignFormat(final ImageInputStream stream) throws IOException {
        byte[] magic = new byte[12]; // Length of JPEG 2000 magic

        try {
            stream.readFully(magic);
        }
        finally {
            stream.seek(0);
        }

        String format;

        if (Arrays.equals(ICNS.PNG_MAGIC, magic)) {
            format = "PNG";
        }
        else if (Arrays.equals(ICNS.JPEG_2000_MAGIC, magic)) {
            format = "JPEG 2000";
        }
        else {
            format = "unknown";
        }

        return format;
    }

    private IconResource readIconResource(final int imageIndex) throws IOException {
        checkBounds(imageIndex);
        readeFileHeader();

        while (icons.size() <= imageIndex) {
            readNextIconResource();
        }

        return icons.get(imageIndex);
    }

    private IconResource readNextIconResource() throws IOException {
        long lastReadPos = lastResourceRead == null ? ICNS.RESOURCE_HEADER_SIZE : lastResourceRead.start + lastResourceRead.length;

        imageInput.seek(lastReadPos);

        if (imageInput.getStreamPosition() >= length) {
            throw new IndexOutOfBoundsException();
        }

        IconResource resource = IconResource.read(imageInput);
//        System.err.println("resource: " + resource);

        lastResourceRead = resource;

        // Filter out special cases like 'icnV' or 'TOC ' resources
        if (resource.isMaskType()) {
            masks.add(resource);
        }
        else if (!resource.isUnknownType()) {
            icons.add(resource);
        }

        return resource;
    }

    private void readeFileHeader() throws IOException {
        assertInput();

        if (length <= 0) {
            imageInput.seek(0);

            if (imageInput.readInt() != ICNS.MAGIC) {
                throw new IIOException("Not an Apple Icon Image");
            }

            length = imageInput.readInt();
        }
    }

    private static final class ICNSBitMaskColorModel extends IndexColorModel {
        static final IndexColorModel INSTANCE = new ICNSBitMaskColorModel();

        private ICNSBitMaskColorModel() {
            super(1, 2, new int[]{0, 0xffffffff}, 0, true, 0, DataBuffer.TYPE_BYTE);
        }
    }

    @SuppressWarnings({"UnusedAssignment"})
    public static void main(String[] args) throws IOException {
        int argIndex = 0;

        int requested = -1;
        if (args[argIndex].charAt(0) == '-') {
            argIndex++;
            requested = Integer.parseInt(args[argIndex++]);
        }

        int imagesRead = 0;
        int imagesSkipped = 0;
        ImageReader reader = new ICNSImageReader();

        while(argIndex < args.length) {
            File input = new File(args[argIndex++]);
            ImageInputStream stream = ImageIO.createImageInputStream(input);

            if (stream == null) {
                System.err.printf("Cannot read: %s\n", input.getAbsolutePath());
                continue;
            }

            try {
                reader.setInput(stream);

                int start = requested != -1 ? requested : 0;
                int numImages = requested != -1 ? requested + 1 : reader.getNumImages(true);
                for (int i = start; i < numImages; i++) {
                    try {
                        long begin = System.currentTimeMillis();
                        BufferedImage image = reader.read(i);
                        imagesRead++;
//                        System.err.println("image: " + image);
                        System.err.println(System.currentTimeMillis() - begin + "ms");
                        showIt(image, String.format("%s - %d", input.getName(), i));
                    }
                    catch (IOException e) {
                        imagesSkipped++;
                        if (e.getMessage().contains("JPEG 2000")) {
                            System.err.printf("%s: %s\n", input, e.getMessage());
                        }
                        else {
                            System.err.printf("%s: ", input);
                            e.printStackTrace();
                        }
                    }
                }
            }
            catch (Exception e) {
                System.err.printf("%s: ", input);
                e.printStackTrace();
            }
        }

        System.err.printf("Read %s images (%d skipped) in %d files\n", imagesRead, imagesSkipped, args.length);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy