3

注:私の質問を完全に理解するには、私の例をコンパイルして実行する必要がある場合があります。これがコーシャでない場合は、事前にお詫び申し上げます。

JToggleButtonおよび に基づく Swing コントロールを作成しようとしていますJPopupMenu

ポップアップ メニューが表示されている場合はトグル ボタンが選択され、ポップアップ メニューが表示されていない場合はトグル ボタンの選択が解除されます。JComboBoxしたがって、ポップアップに任意のコンポーネントを含めることができる点を除いて、動作は に似ています。

次のコードは、コントロールを作成する方法の例です (独自のクラスにあることを除いて... a のようなものですJPopupToggleButton)。残念ながら、異なるルック アンド フィールでは異なる動作を示します (Metal と Nimbus でテストしました)。

ここに投稿されたコードは、Metal では期待どおりに動作しますが、Nimbus では動作しません。Nimbus を使用する場合は、トグル ボタンを繰り返しクリックしてポップアップを表示および非表示にするだけで、意味がわかります。

import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.MouseInfo;
import java.awt.Point;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.*;
import javax.swing.event.PopupMenuEvent;
import javax.swing.event.PopupMenuListener;

public class PopupButtonExample extends JFrame
{
    public static void main( String[] args )
    {
        java.awt.EventQueue.invokeLater( new Runnable()
        {
            @Override
            public void run()
            {
                PopupButtonExample example = new PopupButtonExample();
                example.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
                example.setVisible( true );
            }
        });
    }

    public PopupButtonExample()
    {
        super( "Components in Popup" );

        JPanel popupPanel = new JPanel();
        popupPanel.setLayout( new BorderLayout() );
        popupPanel.add( new JLabel( "This popup has components" ),
                BorderLayout.NORTH );
        popupPanel.add( new JTextArea( "Some text", 15, 20 ),
                BorderLayout.CENTER );
        popupPanel.add( new JSlider(), BorderLayout.SOUTH );

        final JPopupMenu popupMenu = new JPopupMenu();
        popupMenu.add( popupPanel );

        final JToggleButton popupButton = new JToggleButton( "Show Popup" );
        popupButton.addActionListener( new ActionListener()
        {
            @Override
            public void actionPerformed(ActionEvent e)
            {
                if( popupButton.isSelected() )
                    popupMenu.show( popupButton, 0, popupButton.getHeight() );
            }
        });

        popupMenu.addPopupMenuListener( new PopupMenuListener()
        {
            @Override
            public void popupMenuWillBecomeVisible(PopupMenuEvent pme) {}

            @Override
            public void popupMenuCanceled(PopupMenuEvent pme) {}

            @Override
            public void popupMenuWillBecomeInvisible(PopupMenuEvent pme) {
                Point mouseLoc = MouseInfo.getPointerInfo().getLocation();
                Point componentLoc = popupButton.getLocationOnScreen();
                mouseLoc.x -= componentLoc.x;
                mouseLoc.y -= componentLoc.y;
                if( !popupButton.contains( mouseLoc ) )
                    popupButton.setSelected( false );
            }
        });

        JPanel toolBarPanel = new JPanel();
        toolBarPanel.add( popupButton );
        JToolBar toolBar = new JToolBar();
        toolBar.add( toolBarPanel );

        setLayout( new BorderLayout() );
        add( toolBar, BorderLayout.PAGE_START );
        setPreferredSize( new Dimension( 640, 480 ) );
        pack();
    }
}

次の行をコメントアウトすると、コードは Nimbus では期待どおりに動作しますが、Metal では動作しません。繰り返しますが、トグル ボタンをクリックし続けると、私の言いたいことがわかります。

//                Point mouseLoc = MouseInfo.getPointerInfo().getLocation();
//                Point componentLoc = popupButton.getLocationOnScreen();
//                mouseLoc.x -= componentLoc.x;
//                mouseLoc.y -= componentLoc.y;
//                if( !popupButton.contains( mouseLoc ) )

だからここに私の2つの質問があります:

(1) Nimbus で、ポップアップ パネルを非表示にするクリックが、Metal のようにトグル ボタンに渡されないのはなぜですか?

(2) この問題を解決して、すべてのルック アンド フィールで機能するようにするにはどうすればよいですか?

4

2 に答える 2

4
  • Nimbus はバグが多すぎる (そして開発は途中で終了) Metal と比較して、JToggleButton にマウスを 3 回クリックする必要があることがわかります

  • すべての標準的な L&F には、固有の問題、特に SystemLookAndFeel があります。

  • JPopupではなくJWindowを使用してください。JPopupには別のバグもあります。たとえば、JPopupとJCombobox

于 2012-01-18T21:59:10.557 に答える
0

いくつかの調査の後、Nimbus と Metal の違いの原因がわかりました。BasicPopupMenuUI次のフラグは、ポップアップが閉じられたときのイベントの消費を制御するために(少なくとも によって) 使用されます。

UIManager.getBoolean( "PopupMenu.consumeEventOnClose" );

Nimbus を使用している場合、これは を返しますtrue。メタル使用時は を返しますfalse。したがって、メソッドpopupMenuWillBecomeInvisibleは次のように定義する必要があります。

if( UIManager.getBoolean( "PopupMenu.consumeEventOnClose" ) )
{
    popupButton.setSelected( false );
}
else
{
    Point mouseLoc = MouseInfo.getPointerInfo().getLocation();
    Point componentLoc = popupButton.getLocationOnScreen();
    mouseLoc.x -= componentLoc.x;
    mouseLoc.y -= componentLoc.y;
    if( !popupButton.contains( mouseLoc ) )
    {
        popupButton.setSelected( false );
    }
}
于 2012-01-19T18:21:01.793 に答える