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

com.sun.javafx.iio.ImageStorage Maven / Gradle / Ivy

There is a newer version: 24-ea+19
Show newest version
/*
 * Copyright (c) 2009, 2024, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package com.sun.javafx.iio;

import com.sun.javafx.PlatformUtil;
import com.sun.javafx.iio.ImageFormatDescription.Signature;
import com.sun.javafx.iio.bmp.BMPImageLoaderFactory;
import com.sun.javafx.iio.common.ImageTools;
import com.sun.javafx.iio.gif.GIFImageLoaderFactory;
import com.sun.javafx.iio.ios.IosImageLoaderFactory;
import com.sun.javafx.iio.jpeg.JPEGImageLoaderFactory;
import com.sun.javafx.iio.png.PNGImageLoaderFactory;
import com.sun.javafx.logging.PlatformLogger;
import com.sun.javafx.util.DataURI;
import com.sun.javafx.util.Logging;

import java.io.ByteArrayInputStream;
import java.io.EOFException;
import java.io.IOException;
import java.io.InputStream;
import java.io.SequenceInputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map.Entry;

/**
 * A convenience class for simple image loading. Factories for creating loaders
 * for image formats must be registered with this class.
 */
public class ImageStorage {

    /**
     * An enumeration of supported image types.
     */
    public static enum ImageType {

        /**
         * An image with a single channel of 8-bit valued gray levels.
         */
        GRAY,
        /**
         * An image with with two 8-bit valued channels, one of gray levels,
         * the other of non-premultiplied opacity, ordered as GAGAGA...
         */
        GRAY_ALPHA,
        /**
         * An image with with two 8-bit valued channels, one of gray levels,
         * the other of premultiplied opacity, ordered as GAGAGA...
         */
        GRAY_ALPHA_PRE,
        /**
         * An image with with one 8-bit channel of indexes into a 24-bit
         * lookup table which maps the indexes to 8-bit RGB components.
         */
        PALETTE,
        /**
         * An image with with one 8-bit channel of indexes into a 32-bit
         * lookup table which maps the indexes to 8-bit RGBA components
         * wherein the opacity is not-premultiplied.
         */
        PALETTE_ALPHA,
        /**
         * An image with with one 8-bit channel of indexes into a 32-bit
         * lookup table which maps the indexes to 8-bit RGBA components
         * wherein the opacity is premultiplied.
         */
        PALETTE_ALPHA_PRE,
        /**
         * An image with with one 8-bit channel of indexes into a 24-bit
         * lookup table which maps the indexes to 8-bit RGB components, and
         * a single transparent index to indicate the location of transparent
         * pixels.
         */
        PALETTE_TRANS,
        /**
         * An image with with three 8-bit valued channels of red, green, and
         * blue, respectively, ordered as RGBRGBRGB...
         */
        RGB,
        /**
         * An image with with four 8-bit valued channels of red, green, blue,
         * and non-premultiplied opacity, respectively, ordered as
         * RGBARGBARGBA...
         */
        RGBA,
        /**
         * An image with with four 8-bit valued channels of red, green, blue,
         * and premultiplied opacity, respectively, ordered as
         * RGBARGBARGBA...
         */
        RGBA_PRE
    }
//    /**
//     * A mapping of lower case file extensions to loader factories.
//     */
//    private static HashMap loaderFactoriesByExtension;
    /**
     * A mapping of format signature byte sequences to loader factories.
     */
    private final HashMap loaderFactoriesBySignature;
    /**
     * A mapping of lower case MIME subtypes to loader factories.
     */
    private final HashMap loaderFactoriesByMimeSubtype;
    private final ImageLoaderFactory[] loaderFactories;
    private int maxSignatureLength;

    private static final boolean isIOS = PlatformUtil.isIOS();

    private static class InstanceHolder {
        static final ImageStorage INSTANCE = new ImageStorage();
    }

    public static ImageStorage getInstance() {
        return InstanceHolder.INSTANCE;
    }

    public ImageStorage() {
        if (isIOS) {
            //On iOS we have single factory/ native loader
            //for all image formats
            loaderFactories = new ImageLoaderFactory[]{
                IosImageLoaderFactory.getInstance()
            };
        } else {
            loaderFactories = new ImageLoaderFactory[]{
                GIFImageLoaderFactory.getInstance(),
                JPEGImageLoaderFactory.getInstance(),
                PNGImageLoaderFactory.getInstance(),
                BMPImageLoaderFactory.getInstance()
                // Note: append ImageLoadFactory for any new format here.
            };
        }

//        loaderFactoriesByExtension = new HashMap(numExtensions);
        loaderFactoriesBySignature = new HashMap<>(loaderFactories.length);
        loaderFactoriesByMimeSubtype = new HashMap<>(loaderFactories.length);

        for (int i = 0; i < loaderFactories.length; i++) {
            addImageLoaderFactory(loaderFactories[i]);
        }
    }

    public ImageFormatDescription[] getSupportedDescriptions() {
        ImageFormatDescription[] formats = new ImageFormatDescription[loaderFactories.length];
        for (int i = 0; i < loaderFactories.length; i++) {
            formats[i] = loaderFactories[i].getFormatDescription();
        }
        return (formats);
    }

    /**
     * Returns the number of bands for a raw image of the specified type.
     *
     * @param type the type of image
     * @return the number of bands of a raw image of this type
     */
    public int getNumBands(ImageType type) {
        int numBands = -1;
        switch (type) {
            case GRAY:
            case PALETTE:
            case PALETTE_ALPHA:
            case PALETTE_ALPHA_PRE:
            case PALETTE_TRANS:
                numBands = 1;
                break;
            case GRAY_ALPHA:
            case GRAY_ALPHA_PRE:
                numBands = 2;
                break;
            case RGB:
                numBands = 3;
                break;
            case RGBA:
            case RGBA_PRE:
                numBands = 4;
                break;
            default:
                throw new IllegalArgumentException("Unknown ImageType " + type);
        }
        return numBands;
    }

    /**
     * Registers an image loader factory. The factory replaces any other factory
     * previously registered for the file extensions (converted to lower case),
     * MIME subtype, and signature indicated by the format description.
     *
     * @param factory the factory to register.
     */
    public void addImageLoaderFactory(ImageLoaderFactory factory) {
        ImageFormatDescription desc = factory.getFormatDescription();
//        String[] extensions = desc.getExtensions();
//        for (int j = 0; j < extensions.length; j++) {
//            loaderFactoriesByExtension.put(extensions[j].toLowerCase(), factory);
//        }

        for (final Signature signature: desc.getSignatures()) {
            loaderFactoriesBySignature.put(signature, factory);
        }

        for (String subtype : desc.getMIMESubtypes()) {
            loaderFactoriesByMimeSubtype.put(subtype.toLowerCase(), factory);
        }

        // invalidate max signature length
        synchronized (ImageStorage.class) {
            maxSignatureLength = -1;
        }
    }

    /**
     * Load all images present in the specified stream. The image will be
     * rescaled according to this algorithm:
     *
     * 
     * int finalWidth, finalHeight; // final dimensions
     * int width, height;     // specified maximum dimensions
     * // Use source dimensions as default values.
     * if (width <= 0) {
     *     width = sourceWidth;
     * }
     * if (height <= 0) {
     *     height = sourceHeight;
     * }
     * // If not downscaling reset the dimensions to those of the source.
     * if (!((width < sourceWidth && height <= sourceHeight) ||
     *       (width <= sourceWidth && height < sourceHeight))) {
     *      finalWidth = sourceWidth;
     *      finalHeight = sourceHeight;
     * } else if(preserveAspectRatio) {
     *      double r = (double) sourceWidth / (double) sourceHeight;
     *      finalHeight = (int) ((width / r < height ? width / r : height) + 0.5);
     *      finalWidth = (int) (r * finalHeight + 0.5);
     * } else {
     *      finalWidth = width;
     *      finalHeight = height;
     * }
     * 
* * @param input the image data stream. * @param listener a listener to receive notifications about image loading. * @param width the desired width of the image; if non-positive, * the original image width will be used. * @param height the desired height of the image; if non-positive, the * original image height will be used. * @param preserveAspectRatio whether to preserve the width-to-height ratio * of the image. * @param smooth whether to apply smoothing when downsampling. * @return the sequence of all images in the specified source or * null on error. */ public ImageFrame[] loadAll(InputStream input, ImageLoadListener listener, double width, double height, boolean preserveAspectRatio, float pixelScale, boolean smooth) throws ImageStorageException { ImageLoader loader = null; ImageFrame[] images = null; try { if (isIOS) { // no extension/signature recognition done here, // we always want the iOS native loader loader = IosImageLoaderFactory.getInstance().createImageLoader(input); } else { loader = getLoaderBySignature(input, listener); } if (loader != null) { images = loadAll(loader, width, height, preserveAspectRatio, pixelScale, smooth); } else { throw new ImageStorageException("No loader for image data"); } } catch (ImageStorageException ise) { throw ise; } catch (IOException e) { throw new ImageStorageException(e.getMessage(), e); } finally { if (loader != null) { loader.dispose(); } } return images; } /** * Load all images present in the specified input. For more details refer to * {@link #loadAll(InputStream, ImageLoadListener, double, double, boolean, float, boolean)}. */ public ImageFrame[] loadAll(String input, ImageLoadListener listener, double width, double height, boolean preserveAspectRatio, float devPixelScale, boolean smooth) throws ImageStorageException { if (input == null || input.isEmpty()) { throw new ImageStorageException("URL can't be null or empty"); } ImageFrame[] images = null; InputStream theStream = null; ImageLoader loader = null; try { float imgPixelScale = 1.0f; try { DataURI dataUri = DataURI.tryParse(input); if (dataUri != null) { if (!"image".equalsIgnoreCase(dataUri.getMimeType())) { throw new IllegalArgumentException("Unexpected MIME type: " + dataUri.getMimeType()); } // Find a factory that can load images with the specified MIME type. var factory = loaderFactoriesByMimeSubtype.get(dataUri.getMimeSubtype().toLowerCase()); if (factory == null) { throw new IllegalArgumentException( "Unsupported MIME subtype: image/" + dataUri.getMimeSubtype()); } // We also inspect the image file signature to confirm that it matches the MIME type. theStream = new ByteArrayInputStream(dataUri.getData()); ImageLoader loaderBySignature = getLoaderBySignature(theStream, listener); if (loaderBySignature != null) { // If the MIME type doesn't agree with the file signature, log a warning and // continue with the image loader that matches the file signature. boolean imageTypeMismatch = !factory.getFormatDescription().getFormatName().equals( loaderBySignature.getFormatDescription().getFormatName()); if (imageTypeMismatch) { var logger = Logging.getJavaFXLogger(); if (logger.isLoggable(PlatformLogger.Level.WARNING)) { logger.warning(String.format( "Image format '%s' does not match MIME type '%s/%s' in URI '%s'", loaderBySignature.getFormatDescription().getFormatName(), dataUri.getMimeType(), dataUri.getMimeSubtype(), dataUri)); } } loader = loaderBySignature; } else { // We're here because the image format doesn't have a detectable signature. // In this case, we need to close the input stream (because we already consumed // parts of it to detect a potential file signature) and create a new input // stream for the image loader that matches the MIME type. theStream.close(); theStream = new ByteArrayInputStream(dataUri.getData()); loader = factory.createImageLoader(theStream); } } else { // Use Mac Retina conventions for >= 1.5f (rounded to the next integer scale) for (int imageScale = Math.round(devPixelScale); imageScale >= 2; --imageScale) { try { String scaledName = ImageTools.getScaledImageName(input, imageScale); theStream = ImageTools.createInputStream(scaledName); imgPixelScale = imageScale; break; } catch (IOException ignored) { } } if (theStream == null) { theStream = ImageTools.createInputStream(input); } if (isIOS) { loader = IosImageLoaderFactory.getInstance().createImageLoader(theStream); } else { loader = getLoaderBySignature(theStream, listener); } } } catch (Exception e) { throw new ImageStorageException(e.getMessage(), e); } if (loader != null) { images = loadAll(loader, width, height, preserveAspectRatio, imgPixelScale, smooth); } else { throw new ImageStorageException("No loader for image data"); } } finally { if (loader != null) { loader.dispose(); } try { if (theStream != null) { theStream.close(); } } catch (IOException ignored) { } } return images; } private synchronized int getMaxSignatureLength() { if (maxSignatureLength < 0) { maxSignatureLength = 0; for (final Signature signature: loaderFactoriesBySignature.keySet()) { final int signatureLength = signature.getLength(); if (maxSignatureLength < signatureLength) { maxSignatureLength = signatureLength; } } } return maxSignatureLength; } private ImageFrame[] loadAll(ImageLoader loader, double width, double height, boolean preserveAspectRatio, float pixelScale, boolean smooth) throws ImageStorageException { ImageFrame[] images = null; ArrayList list = new ArrayList<>(); int imageIndex = 0; ImageFrame image = null; int imgw = (int) Math.round(width * pixelScale); int imgh = (int) Math.round(height * pixelScale); do { try { image = loader.load(imageIndex++, imgw, imgh, preserveAspectRatio, smooth); } catch (Exception e) { // allow partially loaded animated images if (imageIndex > 1) { break; } else { throw new ImageStorageException(e.getMessage(), e); } } if (image != null) { image.setPixelScale(pixelScale); list.add(image); } else { break; } } while (true); int numImages = list.size(); if (numImages > 0) { images = new ImageFrame[numImages]; list.toArray(images); } return images; } // private static ImageLoader getLoaderByExtension(String input, ImageLoadListener listener) { // ImageLoader loader = null; // // int dotIndex = input.lastIndexOf("."); // if (dotIndex != -1) { // String extension = input.substring(dotIndex + 1).toLowerCase(); // Set extensions = loaderFactoriesByExtension.keySet(); // if (extensions.contains(extension)) { // ImageLoaderFactory factory = loaderFactoriesByExtension.get(extension); // InputStream stream = ImageTools.createInputStream(input); // if (stream != null) { // loader = factory.createImageLoader(stream); // if (listener != null) { // loader.addListener(listener); // } // } // } // } // // return loader; // } private ImageLoader getLoaderBySignature(InputStream stream, ImageLoadListener listener) throws IOException { byte[] header = new byte[getMaxSignatureLength()]; try { ImageTools.readFully(stream, header); } catch (EOFException ignored) { return null; } for (final Entry factoryRegistration: loaderFactoriesBySignature.entrySet()) { if (factoryRegistration.getKey().matches(header)) { InputStream headerStream = new ByteArrayInputStream(header); InputStream seqStream = new SequenceInputStream(headerStream, stream); ImageLoader loader = factoryRegistration.getValue().createImageLoader(seqStream); if (listener != null) { loader.addListener(listener); } return loader; } } // not found return null; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy