提问者:小点点

JFrame中的JPopupMenu使用AWTU的. setWindowOpaque(窗口,false)使用synth L


这个让我很困惑。基本上,我正在使用自定义Synth L使用透明形状的窗口开发一个多窗口应用程序

除了苹果OSX上似乎没有问题之外,我试图把这一点OS细节,但没有太多的乐趣。像我这样的一些视窗用户没有遇到问题,其他人没有……

我还追踪了设置窗口不透明度的违规代码行:

AWTUtilities.setWindowOpaque(window, false) 

如果我删除此LOC则弹出窗口显示良好。此外,将此LOC替换为:

window.setBackground(new Color(0.0f, 0.0f, 0.0f, 0.0f));

产生同样的问题。另一件事是,如果我使用默认的L

只是为了确认JFrameJDialog组件的问题是相同的,只是想知道是否有其他人遇到过这个问题,或者可以为我指出可能的原因。

干杯

要重现的测试源:

import com.sun.awt.AWTUtilities;

import javax.swing.*;
import javax.swing.plaf.synth.SynthLookAndFeel;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

public class TestFrame extends JFrame{

    public TestFrame(){
        super.setTitle("Test Frame");

        JButton btnDialog = new JButton("Open Dialog");
        btnDialog.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                TestDialog dialog = new TestDialog(TestFrame.this, true);
                dialog.setVisible(true);
            }
        });

        super.add(btnDialog, BorderLayout.CENTER);
        super.pack();
        super.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        super.setVisible(true);
    }

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

    public static void initLookAndFeel() {
        SynthLookAndFeel lookAndFeel = new SynthLookAndFeel();
        try {
            lookAndFeel.load(TestFrame.class.getResourceAsStream("/testskin.xml"), TestFrame.class);
            UIManager.setLookAndFeel(lookAndFeel);
        }
        catch (Exception e) {
           e.printStackTrace();
        }
    }

    public static class TestDialog extends JDialog{

        public TestDialog(Frame owner, boolean modal) {
            super(owner, modal);

            JComboBox petList = new JComboBox(new String[] { "Bird", "Cat", "Dog", "Rabbit", "Pig" });
            super.add(petList, BorderLayout.CENTER);

            super.setUndecorated(true);
            AWTUtilities.setWindowOpaque(this, false);
            super.pack();
        }
    }
}

和测试皮肤. xml:

<synth>

    <style id="backingStyle">
        <opaque value="true"/>
        <font name="Dialog" size="14"/>
    </style>
    <bind style="backingStyle" type="region" key=".*"/>

    <style id="ComboBox List Renderer">
        <opaque value="true"/>
        <state value="ENABLED">
            <color type="TEXT_FOREGROUND" value="#000000"/>
        </state>
        <state value="DISABLED">
            <color type="TEXT_FOREGROUND" value="#999999"/>
        </state>
        <state value="SELECTED">
            <color type="TEXT_FOREGROUND" value="#CC6600"/>
            <color type="TEXT_BACKGROUND" value="#FFEEDD"/>
        </state>
    </style>
    <bind style="ComboBox List Renderer" type="name" key="ComboBox.listRenderer" />

    <style id="Combo Box">
        <property key="ComboBox.showPopupOnNavigation" type="boolean" value="true"/>
        <state>
            <color value="#D8D987" type="BACKGROUND"/>
        </state>
    </style>
    <bind style="Combo Box" type="region" key="ComboBox" />

</synth>

如前所述删除:

AWTUtilities.setWindowOpaque(window, false) 

使组合框弹出菜单渲染确定,此外还为所有样式添加默认背景(在style="backingStyle"下),例如:

<state>
   <color value="#D8D987" type="BACKGROUND"/>
</state>

至少会使弹出菜单出现,但是它仍然没有正确渲染。我已经在三个独立的windows xp虚拟机上尝试过这个,都遇到了相同的问题。此外,不要认为我提到过这个,但它是建立在JDK7上的,并且在所有情况下都在等效的JRE上运行。我自己在windows 7终极64位上没有遇到问题,另一个使用windows 7高级64位的用户确实遇到了相同的问题。

一些进展,弹出菜单组件的绘制方法在以下情况下调用失败:

AWTUtilities.setWindowOpaque(window, false) 

设置。在调用“show”方法后手动调用repaint、updateUI、revalate将使弹出菜单呈现ok。对于combox元素设置自定义UI并覆盖“createPopup”方法,使用一个扩展javax. swing.plaf.base.BasicComboPopup的类在show上调用repaint/updateUI/revalate,例如:

 public class ComboPopup extends BasicComboPopup {

    public ComboPopup( JComboBox combo ) {
        super(combo);
    }

    @Override
    public void show(Component invoker, int x, int y) {
        super.show(invoker, x, y);
        this.updateUI();
    }
}

将使组合菜单渲染确定。然而,我还没有找到弹出窗口submneu(JMenu)项目的解决方法,因为弹出窗口是在私有方法中创建的。这似乎是一个bug,但是如果我做错了什么,有人可以让我知道:)

干杯

乔纳森


共1个答案

匿名用户

我在这里回答了一个类似的问题。希望它对找到此线程的其他人有所帮助,我也将在这里提出我的解决方法。

基本上,你的问题是在你需要一个重量级弹出窗口时出现——一个不适合目标窗口的弹出窗口。解决方法是在任何弹出窗口显示后调用重新绘制。只需在启动应用程序时调用以下代码。

PopupFactory.setSharedInstance(new PopupFactory() 
{
    @Override
    public Popup getPopup(Component owner, final Component contents, int x, int y) throws IllegalArgumentException
    {
        Popup popup = super.getPopup(owner, contents, x, y);
        SwingUtilities.invokeLater(new Runnable()
        {
            @Override
            public void run()
            {
                contents.repaint();
            }
        });
        return popup;
    }
});