半透明窗口内的半透明JPopupMenu - 有替代方法吗?

4
我不确定是否有可能,但是是否有一种安全的方法可以允许弹出窗口即使父容器也是半透明的情况下也能保持半透明?如果没有,那么使用或扩展JPopupMenu的明智替代方案是什么?
注意:半透明指组件没有背景,类似于setOpaque(false)的效果。谢谢。
来自2009年用户camickr在论坛上的回答:
我不知道透明度绘画在1.6.0_10中是否有改变。在那之前,我认为透明度只能在轻量级组件(即Swing完成所有绘画)中实现。JFrame、JWindow和JDialog不是轻量级组件,因为它们使用操作系统组件。
对于弹出窗口,在完全包含在其父框架内时,它是轻量级的。但是轻量级弹出窗口无法在框架范围之外绘制,因此使用JWindow作为弹出窗口(我认为)是不透明的。
SSCCE:在半透明JFrame上方显示半透明JWindow。
import com.sun.awt.AWTUtilities;
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.*;

public class OpaqueWindowSSCCE {

    private int countdown = 5;

    public static void main(String[] args) {
        new OpaqueWindowSSCCE();
    }

    public OpaqueWindowSSCCE() {
        final JFrame frame = new JFrame("OpaqueWindowSSCCE");
        final JWindow window = new JWindow();

        new Timer(1000, new ActionListener(){

            @Override
            public void actionPerformed(ActionEvent e) {
                if(--countdown == 0){
                    frame.dispose();
                    window.dispose();
                    System.exit(0);
                } else {
                    frame.repaint();
                }
            }

        }).start();

        frame.setContentPane(new JPanel() {

            @Override
            public void paintComponent(Graphics paramGraphics) {
                super.paintComponent(paramGraphics);
                Graphics2D g = (Graphics2D) paramGraphics.create();
                g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
                g.setColor(new Color(50, 50, 50));
                g.fillRoundRect(0, 0, getWidth(), getHeight(), 10, 10);
                g.setColor(new Color(180, 180, 180));
                g.drawString("Closing in " + countdown + " seconds", 20, 25);
            }
        });

        window.setContentPane(new JPanel() {

            @Override
            public void paintComponent(Graphics paramGraphics) {
                super.paintComponent(paramGraphics);
                Graphics2D g = (Graphics2D) paramGraphics.create();
                g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
                g.setColor(new Color(180, 180, 180));
                g.fillRoundRect(0, 0, getWidth(), getHeight(), 10, 10);
            }
        });

        frame.setUndecorated(true);

        ((JComponent) frame.getContentPane()).setOpaque(false);
        ((JComponent) window.getContentPane()).setOpaque(false);

        AWTUtilities.setWindowOpaque(frame, false);
        AWTUtilities.setWindowOpaque(window, false);

        window.setAlwaysOnTop(true);

        frame.setBounds(200,200,500,500);
        window.setBounds(600,600,200,200);
        frame.setVisible(true);
        window.setVisible(true);
    }
}

我猜在你所做的上下文中,可以使用JWindow的这个函数setOpacity()来实现。 - nIcE cOw
1个回答

4
尝试使用这段代码,我之前用过 JWindow
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;

public class WindowExample
{
    private JWindow window;
    private JLabel updateLabel;
    private int count = 5;
    private Timer timer;
    private int x;
    private int y;
    private ActionListener timerAction = new ActionListener()
    {
        public void actionPerformed(ActionEvent ae)
        {
            updateLabel.setText("Closing Window in " + count + " seconds...");
            count--;
            if (count == 0)
            {
                timer.stop();
                window.setVisible(false);
                window.dispose();
            }   
        }
    };

    private void createAndDisplayGUI()
    {
        final JFrame frame = new JFrame("Window Example");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        frame.setUndecorated(true);
        frame.setOpacity(0.5f);
        frame.addMouseListener(new MouseAdapter()
        {
            public void mouseClicked(MouseEvent me)
            {
                x = me.getX();
                y = me.getY();
                window = new JWindow();
                JPanel contentPane = new JPanel();
                JLabel positionLabel = new JLabel(
                    "X : " + me.getX() + " Y : " + me.getY());
                updateLabel = new JLabel("TImer");  
                contentPane.setLayout(new BorderLayout(5, 5));
                contentPane.add(updateLabel, BorderLayout.CENTER);
                contentPane.add(positionLabel, BorderLayout.PAGE_END);
                window.setContentPane(contentPane);
                window.setOpacity(0.5f);
                window.setSize(200, 100);
                window.setLocation(x + window.getWidth(), y + window.getHeight());
                window.setVisible(true);
                count = 5;
                timer = new Timer(1000, timerAction);
                timer.start();
            }
        });

        frame.setSize(500, 500);
        frame.setLocationRelativeTo(null);
        frame.setVisible(true);
    }

    public static void main(String... args)
    {
        SwingUtilities.invokeLater(new Runnable()
        {
            public void run()
            {
                new WindowExample().createAndDisplayGUI();
            }
        });
    }
}

以下是输出结果:

半透明

警告我正在获取

C:\Mine\JAVA\J2SE>javac -d classes src\OpaqueWindowSSCCE.java
src\OpaqueWindowSSCCE.java:1: warning: AWTUtilities is internal proprietary API and may be removed i
n a future release
import com.sun.awt.AWTUtilities;
                  ^
src\OpaqueWindowSSCCE.java:68: warning: AWTUtilities is internal proprietary API and may be removed
in a future release
        AWTUtilities.setWindowOpaque(frame, false);
        ^
src\OpaqueWindowSSCCE.java:69: warning: AWTUtilities is internal proprietary API and may be removed
in a future release
        AWTUtilities.setWindowOpaque(window, false);
        ^
3 warnings

1
感谢提供SSCCE!很明显JWindow能够正常显示 - 缺点是我需要实现显示和隐藏来模仿JPopupMenu,这并不太难。 - rtheunissen
非常欢迎您,保持微笑 :-),没错,在JPopUpMenu中,我想您必须点击其他地方才能隐藏先前的菜单项 :-),只是JWindow的优点在于我们可以添加任何我们想要展示给用户的内容 :-) - nIcE cOw
是的,但是JPopUpMenu非常好地处理了焦点问题,我们必须确保它能够正常工作。出于某种原因,我完全没有想到使用JWindow...谢谢,我会尝试一下并回来告诉你结果的。 :D - rtheunissen
嗯,这个方法并不起作用,因为设置任意的OPACITY并不是问题所在。我希望弹出窗口根本就没有背景(以便实现圆角效果)。关键要注意的是,通过AWTUtilities.setWindowOpaque(frame, false);可以使JFrame不透明。 - rtheunissen
@paranoid-android:抱歉回复晚了,我们遇到了一些服务器问题。您能否提供一些可行的小代码让我试试手?如果不是很麻烦的话,那就太好了 :-) - nIcE cOw
@paranoid-android:非常感谢你,我可以问你一个简单的问题吗?为什么我会收到这个警告,请检查我的回答。虽然我的回答在某些地方帮助了你,但我很高兴 :-) 我还从你的输入中学到了一些东西,如何绘制圆形的 JWindow ,这太棒了 :-) - nIcE cOw

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接