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

com.github.swingdpi.util.ScaledIcon Maven / Gradle / Ivy

/*
 * Copyright 2016 the original author or authors.
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 2.1 of the License, or
 * (at your option) any later version.
 *
 * This program 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 Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program.  If not, see .
 *
 * This project is hosted at: https://github.com/lukeu/swing-dpi
 * Comments & collaboration are both welcome.
 */

package com.github.swingdpi.util;

import java.awt.Component;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.geom.AffineTransform;
import java.awt.image.AffineTransformOp;
import java.awt.image.BufferedImage;
import java.lang.ref.SoftReference;

import javax.swing.AbstractButton;
import javax.swing.Icon;
import javax.swing.ImageIcon;
import javax.swing.JComponent;
import javax.swing.UIManager;

public class ScaledIcon implements Icon {
    protected final Icon delegate;
    protected final float scaleFactor;
    private final AffineTransformOp scaleOperation;
    private SoftReference enabledIcon = new SoftReference(null);
    private SoftReference disabledIcon = new SoftReference(null);

    public ScaledIcon(Icon icon, float scaleFactor) {

        // Ensure we don't repeatedly scale icons. Callers must have reset the L&F before
        // calling and take care not to scale an icon multiple times, e.g. when iterating the
        // UI-defaults. This can be tricky since UI-defaults may cross-reference each other.
        assert !(icon instanceof ScaledIcon) &&
                !icon.getClass().getSimpleName().contains("ScaledIconUIResource") :
                        "Icon is a:  " + icon.getClass().getName();

        delegate = icon;
        this.scaleFactor = scaleFactor;
        scaleOperation = new AffineTransformOp(
                AffineTransform.getScaleInstance(scaleFactor, scaleFactor),
                AffineTransformOp.TYPE_BICUBIC);
    }

    @Override
    public void paintIcon(Component c, Graphics g, int x, int y) {

        // Weird, this actually happens in Motif. (* roll-eyes *)
        if (getIconWidth() <= 0 || getIconHeight() <= 0) {
            delegate.paintIcon(c, g, x, y);
            return;
        }

        boolean renderEnabled = !(c instanceof AbstractButton) || c.isEnabled();
        ImageIcon icon = renderEnabled ? enabledIcon.get() : disabledIcon.get();
        if (icon == null) {
            icon = new ImageIcon(paintToImageThenScale(c));
            if (delegate instanceof ImageIcon) {
                if (renderEnabled) {
                    enabledIcon = new SoftReference(icon);
                } else {

                    // Note that LookAndFeel#getDisabledIcon only operates upon ImageIcon (despite
                    // having a parameter that takes any Icon). Therefore if 'delegate' is an
                    // ImageIcon we need to render it disabled ourselves, since this class does
                    // not extend ImageIcon.
                    if (c instanceof JComponent) {
                        icon = (ImageIcon) UIManager.getLookAndFeel().getDisabledIcon(
                                (JComponent) c, icon);
                    }
                    disabledIcon = new SoftReference(icon);
                }
            }
        }
        icon.paintIcon(c, g, x, y);
    }

    /**
     * Paints to an image at 100% then performs bicubic scaling to 'scaleFactor'. This approach has
     * consistently produced better quality results. Although some icon painting might use
     * primitive operations (rather than drawing images), it has actually been found to be
     * problematic. For example, the dot in Metal Radio buttons looks 'cracked' at 150% scaling;
     * it appears they draw a Rectangle and 4 lines as an optimisation, and the pieces break
     * apart!
     *
     * TODO: Consider a whitelist of icons that render better directly to a scaled Graphics2D
     * instance? As yet I've not actually seen one, but then I haven't checked through them.
     *
     * UPDATE: huh, this also hit HiDPI in JDK-9: https://bugs.openjdk.java.net/browse/JDK-8160986
     * and looks to have been fixed just a few days ago:
     * http://hg.openjdk.java.net/jdk9/jdk9/jdk/rev/a8d963d7d32d
     */
    private BufferedImage paintToImageThenScale(Component c) {
        BufferedImage image = new BufferedImage(
                getIconWidth(), getIconHeight(), BufferedImage.TYPE_INT_ARGB);
        Graphics2D g2 = null;
        try {

            // While this extra copy may seem like a cost, in ONE **very rough** test (radio buttons
            // in Windows L&F @ 150% scaling) it only added 10% to the mean execution time of this
            // method. Given that, the decision should be more about quality not performance.
            BufferedImage unscaledImage = new BufferedImage(
                    delegate.getIconWidth(), delegate.getIconHeight(), BufferedImage.TYPE_INT_ARGB);
            g2 = unscaledImage.createGraphics();
            delegate.paintIcon(c, g2, 0, 0);
            g2.dispose();

            g2 = image.createGraphics();
            g2.drawImage(unscaledImage, scaleOperation, 0, 0);
        } finally {
            if (g2 != null) {
                g2.dispose();
            }
        }
        return image;
    }

    /*- Alternative rendering method, kept temporarily in case it proves useful...
    private BufferedImage paintToScaledImage(Component c) {
        BufferedImage image =
                new BufferedImage(getIconWidth(), getIconHeight(), BufferedImage.TYPE_INT_ARGB);
        Graphics2D g2 = null;
        try {
            g2 = image.createGraphics();
            g2.scale(scaleFactor, scaleFactor);
            g2.setRenderingHint(
                    RenderingHints.KEY_INTERPOLATION,
                    RenderingHints.VALUE_INTERPOLATION_BICUBIC);
            delegate.paintIcon(c, g2, 0, 0);
        } finally {
            if (g2 != null) {
                g2.dispose();
            }
        }
        return image;
    }
    */

    @Override
    public int getIconWidth() {
        return Math.round(delegate.getIconWidth() * scaleFactor);
    }

    @Override
    public int getIconHeight() {
        return Math.round(delegate.getIconHeight() * scaleFactor);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy