这让我很不解。基本上,我正在使用自定义 Synth L&F 使用透明形状的窗口开发多窗口应用程序。应用程序的部分调用JFrame
/JDialog
来自父框架的组件。在这些组件中,我有弹出菜单和组合框,问题是某些使用应用程序的人在调用时遇到弹出菜单不出现的问题。没有例外,代码执行良好,包括弹出菜单的“显示”方法。
除了在 mac OSX 上似乎没有问题之外,我试图将其确定为操作系统细节,但并没有太多乐趣。像我这样的一些 Windows 用户没有遇到任何问题,而其他人则......
我还追踪了设置窗口不透明度的违规代码行:
AWTUtilities.setWindowOpaque(window, false)
如果我删除了这个 LOC,那么弹出窗口会很好。除了用以下内容替换此 LOC:
window.setBackground(new Color(0.0f, 0.0f, 0.0f, 0.0f));
产生同样的问题。另一件事是,如果我使用默认的 L&F,弹出窗口呈现正常。
只是为了确认两个组件的问题是相同的,JFrame
并且JDialog
只是想知道是否有其他人遇到过这个问题或者可以指出我可能的原因。
干杯
测试源重现:
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();
}
}
}
和testskin.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 虚拟机上试过这个,都遇到同样的问题。此外,我认为我没有提到这一点,但它是基于 JDK 7 构建的,并且在所有情况下都在等效的 JRE 上运行。我自己在 Windows 7 Ultimate 64 位上没有遇到任何问题,另一位使用 Windows 7 Premium 64 位的用户确实遇到了同样的问题。
一些进展,如果出现以下情况,弹出菜单组件的绘制方法无法调用:
AWTUtilities.setWindowOpaque(window, false)
已设置。在调用 'show' 方法后手动调用 repaint、updateUI、revalidate 将使弹出菜单渲染正常。对于设置自定义 UI 并覆盖“createPopup”方法的组合框元素,使用扩展 javax.swing.plaf.basic.BasicComboPopup 的类,该类在显示时调用 repaint/updateUI/revalidate,例如:
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) 项目的解决方法。这似乎是一个错误,但如果我做错了什么,有人可以告诉我:)
干杯
乔纳森