Как запретить отключенному JMenuItem скрывать меню при нажатии? - PullRequest
6 голосов
/ 06 марта 2011

В моем Java-приложении Swing я заметил, что когда я нажимаю на отключенный JMenuItem в JPopupMenu, он скрывает меню, но я не хочу его скрывать, как будто ничего не нажимается. Есть ли способ предотвратить это ?

-----------------------------------> Обновление: добавлен пример кода:

JMenuItem saveMenuItem = new JMenuItem();

saveMenuItem.setEnabled(false);

saveMenuItem.addActionListener(new ActionListener() {
    public void actionPerformed(ActionEvent e) {
        saveMenuItemActionPerformed();
    }
});
add(saveMenuItem);

private void saveMenuItemActionPerformed() {
    System.out.println( "Save clicked." );
}

Ответы [ 7 ]

0 голосов
/ 17 апреля 2011

Это было проверено и работает.

Look & Feel решает, как обрабатывать события мыши на отключенных пунктах меню.В любом случае, вы можете перехватить нежелательные события, используя пользовательский MenuItem.Просто используйте этот код (копировать / вставить):

public class CustomMenuItem extends JMenuItem {

    public CustomMenuItem(String text) {
        super(text);
    }

    public CustomMenuItem() {
        super();
    }

    protected void processMouseEvent(MouseEvent e) {
        if (isEnabled()) super.processMouseEvent(e);
    }
}

Сначала адаптируйте код в соответствии со своими потребностями (необязательно).
Наконец, замените любой JMenuItem наCustomMenuItem.

Вот и все!

0 голосов
/ 28 ноября 2013

Я думаю, что в Java7 это было исправлено.

0 голосов
/ 05 апреля 2011

Единственное решение, которое я мог придумать, для вашей проблемы щелчка по отключению JMenuItem, заставляющего его скрыться, находится ниже:


import java.awt.Color;
import java.awt.Dimension;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import javax.swing.JFrame;
import javax.swing.JMenuItem;
import javax.swing.JPanel;
import javax.swing.JPopupMenu;

public class PopupMenuDisableNoCloseTest extends JPanel implements ActionListener
{
    public static void main(String[] args)
    {
        PopupMenuDisableNoCloseTest p = new PopupMenuDisableNoCloseTest();
        p.setPreferredSize(new Dimension(200, 300));
        p.setBackground(Color.GREEN);
        JPanel contentPane = new JPanel();
        contentPane.add(p);
        final JFrame f = new JFrame();
        final JPopupMenu popup = new JPopupMenu();
        final JMenuItem menuItem1 = new JMenuItem("A popup menu item");
        menuItem1.addActionListener(p);
        menuItem1.addMouseListener(new MouseAdapter()
        {
            @Override
            public void mousePressed(MouseEvent e)
            {
                System.out.println(" menuItem1 mousePressed e.getPoint()=" + e.getPoint());
            }

            @Override
            public void mouseReleased(MouseEvent e)
            {
                System.out.println(" menuItem1 mouseReleased e.getPoint()=" + e.getPoint());
                if(!menuItem1.isEnabled())
                    popup.setVisible(true);
            }
        });
        menuItem1.setEnabled(false);
        popup.add(menuItem1);
        JMenuItem menuItem2 = new JMenuItem("Another popup menu item");
        menuItem2.addActionListener(p);
        popup.add(menuItem2);
        MouseListener popupListener = new PopupListener(popup);
        f.addMouseListener(popupListener);
        f.setContentPane(contentPane);
        f.setSize(800, 600);
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.setVisible(true);
    }

    @Override
    public void actionPerformed(ActionEvent e)
    {
        JMenuItem source = (JMenuItem) (e.getSource());
        String s = "Action event detected. Event source: " + source.getText();
        System.out.println("s=" + s);
    }

    static class PopupListener extends MouseAdapter
    {
        JPopupMenu popup;
        PopupListener(JPopupMenu popupMenu)
        {
            popup = popupMenu;
        }

        @Override
        public void mousePressed(MouseEvent e)
        {
            maybeShowPopup(e);
        }

        @Override
        public void mouseReleased(MouseEvent e)
        {
            maybeShowPopup(e);
        }

        private void maybeShowPopup(MouseEvent e)
        {
            if(e.isPopupTrigger())
            {
                popup.show(e.getComponent(),
                        e.getX(), e.getY());
            }
        }
    }
}

По сути, скрытие происходит, когда ваш релиз находится в пределах границJMenuItem, поэтому мы проверяем, отключен ли он, и снова показываем всплывающее окно.Поскольку к этому времени уже решено, что это будет скрыто.Я пытался вызвать super.mouseRelease с другим MouseEvent, указывающим на внешний компонент и использующим предыдущий, но это ничего не помогает.

В любом случае это решение работает.Наслаждайся, Боро

0 голосов
/ 10 марта 2011

Вы попробовали этот метод: http://download.oracle.com/javase/6/docs/api/javax/swing/JMenuItem.html#setArmed%28boolean%29

«поставить на охрану пункт меню, чтобы его можно было выбрать», что, я думаю, сработает, если будет установлено значение false.

0 голосов
/ 08 марта 2011

Короче говоря, вы можете сделать это, но вам придется написать свой собственный слушатель мыши, который может потребовать много копировать и вставлять из исходного кода jdk, что не очень хорошая идея, и я не уверен насчеткакие лицензионные ограничения он наложит на ваш код.

Я бы начал копать с этого метода:

javax.swing.plaf.basic.BasicMenuItemUI.Handler#mouseReleased

, который кажется точкой входа, откуда механизм обработки меню скрывает всплывающее окно.Я бы посмотрел поближе на

javax.swing.plaf.basic.BasicPopupMenuUI.MouseGrabber#stateChanged

EDIT Разработка ответа от @Burhan Valikarimwala, попробуйте это apporach: удалите всех слушателей действий из отключенного JMenuItem и сохраните их в некоторой статической временной структуре(скажем, Map<WeakReference<JMenuItem>, List<MouseListener>>), таким образом он не будет скрывать всплывающее окно.Когда вы снова включите пункт меню, добавьте всех слушателей обратно.Сделайте это в некотором методе util, и он будет без шва.

0 голосов
/ 08 марта 2011

Когда вы отключаете JMenuItem, вы должны удалить ActionListener, связанный с этим JMenuItem, используя метод jMenuItem.removeActionListener ().Если вы удалите это действие, это не вызовет слушателя, и всплывающее окно не исчезнет.Я надеюсь, что это поможет достичь вашей цели.

0 голосов
/ 07 марта 2011

не уверен, как предотвратить. но вы можете установить Visible (false), чтобы предотвратить его отображение. Также, если пользователь нажимает на меню отключения, никаких действий не происходит.

...