9

我在 Swing JToolBar 上创建了一个下拉菜单。但它并没有按照我想要的方式创建行为。我的目标是让它像 Firefox 的“智能书签”按钮一样工作。

当用户选择一个菜单项时它会消失:正确!

当用户按下 ESC 时它消失:正确!

当用户单击菜单外主框架中的某个位置时,它会消失:正确!

但是当用户第二次单击显示下拉菜单的按钮时它不会消失:不正确... :-(

我的问题是如何添加此行为,当单击第二次显示菜单的按钮时它会消失。

这是我当前的代码,来自 Mac 上的 Java 6:

import javax.swing.*;
import javax.swing.event.PopupMenuEvent;
import javax.swing.event.PopupMenuListener;
import java.awt.*;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;

public class ScratchSpace {

    public static void main(String[] arguments) {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                JFrame frame = new JFrame("Toolbar with Popup Menu demo");

                final JToolBar toolBar = new JToolBar();
                toolBar.add(createMoreButton());

                final JPanel panel = new JPanel(new BorderLayout());
                panel.add(toolBar, BorderLayout.NORTH);
                panel.setPreferredSize(new Dimension(600, 400));
                frame.getContentPane().add(panel);
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    private static AbstractButton createMoreButton() {
        final JToggleButton moreButton = new JToggleButton("More...");
        moreButton.addItemListener(new ItemListener() {
            public void itemStateChanged(ItemEvent e) {
                if (e.getStateChange() == ItemEvent.SELECTED) {
                    createAndShowMenu((JComponent) e.getSource(), moreButton);
                }
            }
        });
        moreButton.setFocusable(false);
        moreButton.setHorizontalTextPosition(SwingConstants.LEADING);
        return moreButton;
    }

    private static void createAndShowMenu(final JComponent component, final AbstractButton moreButton) {
        JPopupMenu menu = new JPopupMenu();
        menu.add(new JMenuItem("Black"));
        menu.add(new JMenuItem("Red"));

        menu.addPopupMenuListener(new PopupMenuListener() {
            public void popupMenuWillBecomeVisible(PopupMenuEvent e) {
            }

            public void popupMenuWillBecomeInvisible(PopupMenuEvent e) {
                moreButton.setSelected(false);
            }

            public void popupMenuCanceled(PopupMenuEvent e) {
                moreButton.setSelected(false);
            }
        });

        menu.show(component, 0, component.getHeight());
    }
}
4

4 回答 4

5

好吧,这是一个潜在的解决方案,它并非没有缺点。只有您可以决定这是否适合您的应用程序。问题是弹出窗口关闭发生在触发其他鼠标处理事件之前,因此再次单击您的 More.. 按钮会导致弹出窗口隐藏,从而在按钮甚至被告知它被按下之前将按钮状态重置为取消选择。

简单的解决方法是在主程序中添加以下调用:

UIManager.put("PopupMenu.consumeEventOnClose", Boolean.TRUE);

这样做的结果是,每当弹出菜单由于鼠标按下事件而关闭时,该鼠标事件将在菜单关闭时被消耗,并且不会传递给鼠标下的任何其他组件。如果您可以忍受限制,这是一个简单的解决方案。

于 2009-12-17T17:05:59.767 回答
1

发生的情况是,当您单击菜单时,它会取消弹出菜单,因此您取消选择该按钮,但下一个即时事件是单击该按钮,现在它已取消选择,因此它再次显示菜单。

我还没有确切的解决方案,但给我一点...

于 2009-12-17T17:18:58.617 回答
0

我不使用 Firefox,所以我不知道智能书签按钮的外观,但也许使用 JMenu 作为“按钮”。您可以尝试使用 JButton 的 Border 使其看起来更像一个按钮。

于 2009-12-14T16:05:19.373 回答
-1

好吧,按钮上的侦听器仅在按下按钮时才做出反应,因为您ItemEvent.SELECTED只侦听事件。如何在ItemEvent.DESELECTED此处添加另一个 if 子句来监听事件:

    moreButton.addItemListener(new ItemListener() {
        public void itemStateChanged(ItemEvent e) {
            if (e.getStateChange() == ItemEvent.SELECTED) {
                createAndShowMenu((JComponent) e.getSource(), moreButton);
            }
        }
    });

您可以存储对menu某处的引用,可以让菜单本身向按钮添加另一个侦听器。后一种解决方案可能更直接,因为您似乎已经向菜单发送了一个按钮引用。

于 2009-12-14T11:21:21.220 回答