如何在非透明JFrame中正确使用透明JPanel?

时间:2014-01-02 04:13:02

标签: java swing jpanel transparent

我正在开发一些不是100%不透明的应用程序,所以它基本上会使用户的桌面变暗,而我的Swing界面显示在这个“黑暗面纱”的顶部。

在我看来,当一些Swing组件移动到那个面纱上时,我的JFrame需要重新绘制我的移动组件,而不是留下它们后面的痕迹。问题是重新粉刷JFrame 太慢,我的应用程序将无法顺利运行。

为方便起见,我创建了一个SSCCE类来说明我的问题,这里是:

import java.awt.Color;
import java.awt.Dimension;
import java.awt.Toolkit;

import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;

@SuppressWarnings("serial")
public class TransparentFrameSSCCE extends JFrame {
    private static final Dimension SCREEN_DIMENSIONS = Toolkit.getDefaultToolkit().getScreenSize();
    private final JPanel movingPanel;

    private TransparentFrameSSCCE() {
        super();

        this.setUndecorated(true);
        this.setResizable(false);

        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.setSize(TransparentFrameSSCCE.SCREEN_DIMENSIONS);
            // This makes my JFrame transparent (its alpha component is set to 0)
        this.setBackground(new Color(0, 0, 0, 0));

        this.movingPanel = new JPanel();
        this.movingPanel.setBounds(0, 0, 50, 50);
        this.movingPanel.setBackground(Color.RED);

        final JPanel contentPane = new JPanel();
            // This makes my panel semi-transparent (its alpha component is set to 128)
        contentPane.setBackground(new Color(0, 0, 0, 128));
        contentPane.setLayout(null);
        contentPane.add(this.movingPanel);

        this.setContentPane(contentPane);
    }

    @Override
    public void setVisible(final boolean isVisible) {
        super.setVisible(isVisible);

        new Thread(new Runnable() {
            @Override
            public void run() {
                int x, y;

                for(;;) {
                    x = TransparentFrameSSCCE.this.movingPanel.getLocation().x;
                    y = TransparentFrameSSCCE.this.movingPanel.getLocation().y;
                    TransparentFrameSSCCE.this.movingPanel.setLocation(x + 5, y);

                    try {
                        Thread.sleep(10);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }).start();
    }

    public static void main(final String args[]) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                new TransparentFrameSSCCE().setVisible(true);
            }
        });
    }
}

有人会知道其他任何方式吗?

更新:关注@MadProgrammer关于Swing组件透明度行为的指示,这就是如何处理我的“黑暗面纱”。它完美地运作。非常感谢他:)

    final JPanel contentPane = new JPanel() {
        @Override
        protected void paintComponent(Graphics g) {
            super.paintComponent(g);
            final Graphics2D g2d = (Graphics2D) g.create();
            g2d.setColor(new Color(0, 0, 0, 128));
            g2d.fill(new Area(new Rectangle(new Point(0, 0), getSize())));
            g2d.dispose();
        }
    };

    contentPane.setOpaque(false);    // Instead of: contentPane.setColor(new Color(0, 0, 0, 128)

2 个答案:

答案 0 :(得分:3)

Java组件没有透明度的概念,它们是不透明的或完全透明的(好吧,顶级窗口的新透明支持是一个例外;)

您需要做的是创建一个完全透明的自定义组件,并覆盖它paintComponent并用半透明颜色填充组件区域。

另外,不要在Event Dispatching Thread的上下文之外修改Swing组件的状态,奇怪的事情开始发生。更好的解决方案可能是使用javax.swing.Timer

例如

您可能还想查看Concurrency in Swing

答案 1 :(得分:3)

查看Backgrounds With Transparency以获取问题的简单说明。基本上,您需要确保自定义组件描绘背景。

或者不用做自定义绘画,你可以利用AlphaContainer类为你做绘画:

//this.setContentPane( contentPane);
this.setContentPane( new AlphaContainer(contentPane) );