JPopupMenu в JFrame с использованием AWTUtilities.setWindowOpaque(window, false) с использованием синтезатора L&F не отображается
Это меня довольно озадачило. По сути, я занимаюсь разработкой многооконного приложения с использованием прозрачных окон с использованием пользовательских Synth L&F. Части приложения вызывают JFrame
/JDialog
компоненты из родительского кадра. В этих компонентах у меня есть всплывающие меню и комбинированные списки, проблема в том, что некоторые люди, использующие приложения, сталкиваются с тем, что всплывающие меню не появляются при вызове. Исключений нет, и код выполняется нормально, в том числе для всплывающих меню, метод show.
Я попытался прибавить это к специфике ОС без особой радости, за исключением того, что, похоже, нет проблем с 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. Я сам на 64-битной Windows 7 Ultimate не испытывают проблем, другой пользователь, использующий 64-битную Windows 7 Premium, испытывает те же проблемы.
Некоторый прогресс, метод рисования для компонентов всплывающего меню не может быть вызван, если:
AWTUtilities.setWindowOpaque(window, false)
установлено. Вызов вручную repaint, updateUI, revalidate после вызова метода 'show' заставит всплывающее меню отображаться нормально. Для элементов combox, устанавливающих пользовательский интерфейс и переопределяющих метод 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();
}
}
сделает меню combox рендеринга нормально. Однако мне еще предстоит найти обходной путь для элементов submneu (JMenu) для всплывающего окна, так как оно создается в закрытом методе. Это похоже на ошибку, но если я делаю что-то не так, может кто-нибудь сообщить мне:)
ура
Джонатан
1 ответ
Я ответил на аналогичный вопрос здесь. Надеюсь, что это будет полезно для других, кто найдет эту ветку, я также опубликую свое решение здесь.
По сути, ваша проблема возникает всякий раз, когда вам нужен HeavyWeightPopup - всплывающее окно, которое не помещается в целевом окне. Обходной путь должен вызвать перерисовку после любых всплывающих окон. Просто запустите следующий код при запуске приложения.
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;
}
});