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;
    }
});
Другие вопросы по тегам