这个让我很困惑。基本上,我正在使用自定义Synth L使用透明形状的窗口开发一个多窗口应用程序
除了苹果OSX上似乎没有问题之外,我试图把这一点OS细节,但没有太多的乐趣。像我这样的一些视窗用户没有遇到问题,其他人没有……
我还追踪了设置窗口不透明度的违规代码行:
AWTUtilities.setWindowOpaque(window, false)
如果我删除此LOC则弹出窗口显示良好。此外,将此LOC替换为:
window.setBackground(new Color(0.0f, 0.0f, 0.0f, 0.0f));
产生同样的问题。另一件事是,如果我使用默认的L
只是为了确认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();
}
}
}
和测试皮肤. 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,但是如果我做错了什么,有人可以让我知道:)
干杯
乔纳森
我在这里回答了一个类似的问题。希望它对找到此线程的其他人有所帮助,我也将在这里提出我的解决方法。
基本上,你的问题是在你需要一个重量级弹出窗口时出现——一个不适合目标窗口的弹出窗口。解决方法是在任何弹出窗口显示后调用重新绘制。只需在启动应用程序时调用以下代码。
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;
}
});