Java Swing:按钮仅在鼠标悬停时显示图标

Java Swing:按钮仅在鼠标悬停时显示图标,java,swing,icons,Java,Swing,Icons,我有一个Swing应用程序,我想在JButton上显示一个图标。我在Windows缩放图标并使其非常难看时遇到了一些问题。我能够通过使用如上所述的图标包装器类来解决这个问题 使用包装器类解决了我遇到的缩放问题,但带来了一个新问题:图标仅在我将鼠标移到按钮上时显示,如图所示: 下面是一个简单的例子: import javax.swing.*; import java.awt.*; import java.net.MalformedURLException; import java.net.URL

我有一个Swing应用程序,我想在JButton上显示一个图标。我在Windows缩放图标并使其非常难看时遇到了一些问题。我能够通过使用如上所述的图标包装器类来解决这个问题

使用包装器类解决了我遇到的缩放问题,但带来了一个新问题:图标仅在我将鼠标移到按钮上时显示,如图所示:

下面是一个简单的例子:

import javax.swing.*;
import java.awt.*;
import java.net.MalformedURLException;
import java.net.URL;

public class Ui {
    private static JFrame frame;
    private static JButton button1;
    private static JButton button2;
    private static JMenuBar jMenuBar;


    public static void main(String[] args) {
        SwingUtilities.invokeLater(() -> {
            try {
                createGui();
            } catch (MalformedURLException e) {
                e.printStackTrace();
            }
        });
    }

    private static void createGui() throws MalformedURLException {
        frame = new JFrame("Test");

        button1 = new JButton();
        button2 = new JButton();
        jMenuBar = new JMenuBar();


        Icon icon = new NoScalingIcon(new ImageIcon(new URL("https://i.stack.imgur.com/wgKeq.png")));
        button1.setIcon(icon);
        button2.setIcon(icon);

        jMenuBar.add(button1);
        jMenuBar.add(button2);
        frame.setJMenuBar(jMenuBar);
        frame.pack();
        frame.setVisible(true);
        frame.setExtendedState(JFrame.MAXIMIZED_BOTH);
    }
}
图标包装类:

import javax.swing.*;
import java.awt.*;
import java.awt.geom.AffineTransform;

public class NoScalingIcon implements Icon {
    private Icon icon;

    public NoScalingIcon(Icon icon) {
        this.icon = icon;
    }

    public int getIconWidth() {
        return icon.getIconWidth();
    }

    public int getIconHeight() {
        return icon.getIconHeight();
    }

    public void paintIcon(Component c, Graphics g, int x, int y) {
        Graphics2D g2d = (Graphics2D) g.create();

        AffineTransform at = g2d.getTransform();
        int scaleX = (int) (x * at.getScaleX());
        int scaleY = (int) (y * at.getScaleY());

        int offsetX = (int) (icon.getIconWidth() * (at.getScaleX() - 1) / 2);
        int offsetY = (int) (icon.getIconHeight() * (at.getScaleY() - 1) / 2);

        g2d.setTransform(new AffineTransform());

        icon.paintIcon(c, g2d, scaleX + offsetX, scaleY + offsetY);

        g2d.dispose();
    }
}

这个问题的原因是什么?

我想我找到了解决办法

下面的代码不是创建一个全新的仿射变换,而是将一个新的缩放变换与现有比例的倒数合并,以便将比例因子设置回1:

import java.awt.*;
import java.awt.geom.AffineTransform;
import javax.swing.*;


public class NoScalingIcon implements Icon
{
    private Icon icon;

    public NoScalingIcon(Icon icon)
    {
        this.icon = icon;
    }

    public int getIconWidth()
    {
        return icon.getIconWidth();
    }

    public int getIconHeight()
    {
        return icon.getIconHeight();
    }

    public void paintIcon(Component c, Graphics g, int x, int y)
    {
        Graphics2D g2d = (Graphics2D)g.create();

        AffineTransform at = g2d.getTransform();

        int scaleX = (int)(x * at.getScaleX());
        int scaleY = (int)(y * at.getScaleY());

        int offsetX = (int)(icon.getIconWidth() * (at.getScaleX() - 1) / 2);
        int offsetY = (int)(icon.getIconHeight() * (at.getScaleY() - 1) / 2);

        int locationX = scaleX + offsetX;
        int locationY = scaleY + offsetY;

        AffineTransform scaled = AffineTransform.getScaleInstance(1.0 / at.getScaleX(), 1.0 / at.getScaleY());
        at.concatenate( scaled );
        g2d.setTransform( at );

        icon.paintIcon(c, g2d, locationX, locationY);

        g2d.dispose();
    }
}

在修复图标后,您可能只需要调用repaint()。@ControlAltDel Ok我尝试在框架上调用repaint(),并在main的最后一行下调用按钮(因此在frame.setExtendedState(JFrame.MAXIMIZED_BOTH);),但这并没有改变任何事情。我想这就是你所说的“call repaint()”工作得很好,非常感谢:)