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

org.openide.util.CachedHiDPIIcon Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.openide.util;

import java.awt.Component;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GraphicsConfiguration;
import java.awt.Image;
import java.awt.Transparency;
import java.awt.geom.AffineTransform;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Objects;
import javax.swing.Icon;

/**
 * Abstract base class for {@link javax.swing.Icon} implementations that need to cache scaled bitmap
 * representations for HiDPI displays. Bitmaps for multiple HiDPI scaling factors can be cached at
 * the same time, e.g. for multi-monitor setups. Thread-safe.
 *
 * @since 9.15
 */
public abstract class CachedHiDPIIcon implements Icon {
    /**
     * The maximum size of the cache, as a multiple of the size of the icon at 100% scaling. For
     * example, storing three images at 100%, 150%, and 200% scaling, respectively, yields a total
     * cache size of 1.0^2 + 1.5^2 + 2^2 = 7.2.
     */
    private static final double MAX_CACHE_SIZE = 10.0;
    private final int width;
    private final int height;
    /**
     * Cache map with least-recently-used iteration order.
     */
    private final Map cache =
            new LinkedHashMap(16, 0.75f, true);
    /**
     * Total size of the images currently in the cache, in the same units as
     * {@link #MAX_CACHE_SIZE}.
     */
    private double cacheSize = 0.0;

    /**
     * Constructor to be used by subclasses. For HiDPI screens, dimensions are specified in logical
     * pixels rather than device pixels, as in the superclass.
     *
     * @param width the width of the icon
     * @param height the height of the icon
     */
    protected CachedHiDPIIcon(int width, int height) {
        if (width < 0) {
            throw new IllegalArgumentException();
        }
        if (height < 0) {
            throw new IllegalArgumentException();
        }
        this.width = width;
        this.height = height;
    }

    /**
     * Get a scaled bitmap image of this icon. This method may not be called if either of the icon's
     * dimensions are zero.
     */
    private synchronized Image getScaledImageCached(Component c, CachedImageKey key) {
        Image ret = cache.get(key);
        if (ret != null) {
            return ret;
        }
        final double scale = key.getScale();
        final int deviceWidth = (int) Math.ceil(getIconWidth() * scale);
        final int deviceHeight = (int) Math.ceil(getIconHeight() * scale);
        final Image img =
                createAndPaintImage(c, key.getColorModel(), deviceWidth, deviceHeight, scale);
        final double imgSize = key.getSize();
        if (imgSize <= MAX_CACHE_SIZE) {
            /* Evict least-recently-used images from the cache until we have space for the latest
            image. */
            final Iterator iter = cache.keySet().iterator();
            while (cacheSize + imgSize > MAX_CACHE_SIZE && iter.hasNext()) {
                CachedImageKey removeKey = iter.next();
                iter.remove();
                cacheSize -= removeKey.getSize();
            }
            cache.put(key, img);
            cacheSize += imgSize;
        }
        return img;
    }

    @Override
    public final void paintIcon(Component c, Graphics g0, int x, int y) {
        if (getIconWidth() == 0 || getIconHeight() == 0) {
            // Special case: Avoid attempting to create a zero-width/height image.
            return;
        }
        final Graphics2D g = (Graphics2D) g0;
        CachedImageKey key = CachedImageKey.create(g);
        final AffineTransform oldTransform = g.getTransform();
        try {
            g.translate(x, y);
            Image scaledImage = getScaledImageCached(c, key);
            /* Scale the image down to its logical dimensions, then draw it at the device pixel
            boundary. In VectorIcon, we tried to be a lot more conservative, taking great care not
            to draw on any device pixels that were only partially bounded by the icon (due to
            non-integral scaling factors, e.g. 150%). That was probably overkill; it's a lot easier
            to assume that partially bounded pixels are OK to draw on, since all icon bitmaps of a
            given scaling factor then end up being the same number of device pixels wide and tall.
            And we need consistent dimensions to be able keep cached images in any case. For these
            reasons, round the X and Y translations (which denote the position in device pixels)
            _down_ here.*/
            AffineTransform tx2 = g.getTransform();
            g.setTransform(new AffineTransform(1, 0, 0, 1,
                    (int) tx2.getTranslateX(),
                    (int) tx2.getTranslateY()));
            g.drawImage(scaledImage, 0, 0, null);
        } finally {
            g.setTransform(oldTransform);
        }
    }

    @Override
    public final int getIconWidth() {
        return width;
    }

    @Override
    public final int getIconHeight() {
        return height;
    }

    /**
     * Create a scaled image containing the graphics of this icon. The result may be cached. The
     * dimensions are specfied in device pixels rather than logical pixels, i.e. with HiDPI scaling
     * applied.
     *
     * @param c the component that was passed to {@link Icon#paintIcon(Component,Graphics,int,int)}.
     *        The cache will not be invalidated if {@code c} or its state changes, so 
     *        subclasses should avoid depending on it if possible. This parameter exists mainly to
     *        ensure compatibility with existing Icon implementations that may be used as delegates.
     *        Future implementations might also elect to simply pass a dummy Component instance
     *        here.
     * @param colorModel the {@link ColorModel} of the surface on which the image will be painted
     *        (may be passed to {@link #createBufferedImage(ColorModel, int, int)} in the common
     *        case)
     * @param deviceWidth the required width of the image, in device pixels (>=1)
     * @param deviceHeight the required height of the image, in device pixels (>=1)
     * @param scale the HiDPI scaling factor detected in {@code graphicsConfiguration}
     */
    protected abstract Image createAndPaintImage(Component c, ColorModel colorModel,
            int deviceWidth, int deviceHeight, double scale);

    /**
     * Utility method to create a compatible {@link BufferedImage} from the parameters passed to
     * {@link #createAndPaintImage(Component, ColorModel, int, int, double)}. May be called by
     * implementors of the latter to create a surface to draw on and return.
     *
     * @param colorModel the required {@link ColorModel}
     * @param deviceWidth the required width of the image, in device pixels (>=1)
     * @param deviceHeight the required height of the image, in device pixels (>=1)
     * @return an image compatible with the given parameters
     */
    protected static final BufferedImage createBufferedImage(
            ColorModel colorModel, int deviceWidth, int deviceHeight)
    {
      return new BufferedImage(
          colorModel,
          colorModel.createCompatibleWritableRaster(deviceWidth, deviceHeight),
          colorModel.isAlphaPremultiplied(), null);
    }

    /**
     * Key for image cache map. Immutable.
     */
    private static final class CachedImageKey {
        /* The ColorModel is the only field in GraphicsConfiguration that is needed to create a
        compatible BufferedImage. So include only that one, plus the HiDPI scaling factor, in the
        cache key. */
        private final ColorModel colorModel;
        private final double scale;

        private CachedImageKey(ColorModel colorModel, double scale) {
            Parameters.notNull("colorModel", colorModel);
            if (scale <= 0.0) {
                throw new IllegalArgumentException();
            }
            this.colorModel = colorModel;
            this.scale = scale;
        }

        public static CachedImageKey create(Graphics2D g) {
            final AffineTransform tx = g.getTransform();
            final int txType = tx.getType();
            final double scale;
            if (txType == AffineTransform.TYPE_UNIFORM_SCALE ||
                txType == (AffineTransform.TYPE_UNIFORM_SCALE | AffineTransform.TYPE_TRANSLATION))
            {
                scale = tx.getScaleX();
            } else {
                scale = 1.0;
            }
            GraphicsConfiguration gconf = g.getDeviceConfiguration();
            /* Always use the same transparency mode for the cached images, so we don't end up with
            one set of cached images for each encountered mode. The TRANSLUCENT mode is the most
            generic one; presumably it should paint OK onto less capable surfaces. */
            ColorModel colorModel = gconf.getColorModel(Transparency.TRANSLUCENT);
            return new CachedImageKey(colorModel, scale);
        }

        public double getScale() {
            return scale;
        }

        /**
         * Get the size of this image as a multiple of the original image's size at 100% scaling.
         */
        public double getSize() {
            return Math.pow(getScale(), 2.0);
        }

        public ColorModel getColorModel() {
            return colorModel;
        }

        @Override
        public int hashCode() {
            return Objects.hash(colorModel, scale);
        }

        @Override
        public boolean equals(Object obj) {
            if (!(obj instanceof CachedImageKey)) {
                return false;
            }
            final CachedImageKey other = (CachedImageKey) obj;
            return this.colorModel.equals(other.colorModel) &&
                   this.scale == other.scale;
        }

        @Override
        public String toString() {
            return "CachedImageKey(" + colorModel + ", " + scale + ")";
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy