7

过去,当一个JPopupMenu可见时,默认情况下会选择第一项:http ://weblogs.java.net/blog/alexfromsun/archive/2008/02/jtrayicon_updat.html

如今,默认行为是弹出菜单而不选择任何项目。我想创建一个JPopupMenu带有单个项目的项目,该项目将在鼠标指针下弹出并居中。我设法让该项目在鼠标下方居中弹出,但我JMenuItem拒绝渲染,就好像它被选中一样。如果我将鼠标移出项目并重新选择正确。

有任何想法吗?

这是我的测试用例:

import java.awt.Component;
import java.awt.Point;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import javax.swing.JFrame;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;

public class Test extends JFrame
{
    public static void main(String[] args)
    {
        JFrame frame = new JFrame();
        frame.setSize(800, 600);
        frame.getContentPane().addMouseListener(new MouseAdapter()
        {
            @Override
            public void mousePressed(MouseEvent e)
            {
                if (e.isPopupTrigger())
                    popupTriggered(e);
            }

            @Override
            public void mouseReleased(MouseEvent e)
            {
                if (e.isPopupTrigger())
                    popupTriggered(e);
            }

            private void popupTriggered(MouseEvent e)
            {
                JPopupMenu menu = new JPopupMenu();
                final JMenuItem item = new JMenuItem("This is a JMenuItem");
                menu.add(item);
                Point point = e.getPoint();
                int x = point.x - (item.getPreferredSize().width / 2);
                int y = point.y - (item.getPreferredSize().height / 2);
                menu.show((Component) e.getSource(), x, y);
            }
        });
        frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
        frame.setVisible(true);
    }
}
4

4 回答 4

8

秘密原来是MenuSelectionManager.defaultManager().setSelectedPath(new MenuElement[]{menu, ...});

import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import javax.swing.JFrame;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;
import javax.swing.MenuElement;
import javax.swing.MenuSelectionManager;
import javax.swing.SwingUtilities;

/**
 * Demonstrates programmatic {@link JMenuItem} selection;
 * specifically how to make the first item selected by default
 */
public class TestPopup extends JFrame {
  public static void main(String[] args) {
    final JFrame frame = new JFrame("TestPopup");
    frame.setSize(640, 480);
    frame.getContentPane().addMouseListener(new MouseAdapter() {
      @Override
      public void mousePressed(MouseEvent e) {
        if (e.isPopupTrigger()) {
          popupTriggered(e);
        }
      }
      private void popupTriggered(MouseEvent e) {
        final JPopupMenu menu = new JPopupMenu();
        final JMenuItem item0 = new JMenuItem("JMenuItem 0");
        final JMenuItem item1 = new JMenuItem("JMenuItem 1");
        menu.add(item0);
        menu.add(item1);
        menu.pack();
        // use invokeLater or just do this after the menu has been shown
        SwingUtilities.invokeLater(new Runnable() {
          public void run() {
            MenuSelectionManager.defaultManager().setSelectedPath(new MenuElement[]{menu, item0});
          }
        });
        int x = (int) ((int) (frame.getSize().width - (menu.getPreferredSize().width / 2.)) / 2.);
        int y = (int) ((int) (frame.getSize().height - (menu.getPreferredSize().height / 2.)) / 2.);
        menu.show(frame, x, y);
        // doesn't work:
        //item0.setSelected(true);
        // doesn't work:
        //menu.getSelectionModel().setSelectedIndex(0);
        // bingo; see also MenuKeyListener / MenuKeyEvent
//        MenuSelectionManager.defaultManager().setSelectedPath(new MenuElement[]{menu, item0});
      }
    });
    frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
    frame.setLocationRelativeTo(null);
    frame.setVisible(true);
  }
}
于 2009-11-27T23:30:51.863 回答
1

如今,默认行为是弹出菜单而不选择任何项目。

实际上,我认为这是正确的行为,至少在 Windows 中是这样。其他非 Java 应用程序也这样做。即使菜单中只有一个项目,我认为不值得打破这个约定。如果您不这么认为,您可以按照sean.bright 的回答设置选择索引。


所以,我终于有机会在 Java 上尝试了一下1.6.0_11,发现了一些不一致的行为:如果弹出菜单突出父框架,则自动选择该项目;如果弹出菜单完全出现在父框架内,则不选择任何内容。听起来像一个 Swing 错误,它至少保证了 RFE 的一致行为。

于 2009-01-27T01:40:25.940 回答
1

MenuSelectionManager.defaultManager() 确实是一个很好的解决方案,但是当您尝试预先选择 JPopupMenu 的子菜单时(它会隐藏父菜单),它就不起作用了。此外,它会扰乱其他键盘导航行为(您不能按左键隐藏子菜单等)

不幸的是,在 Swing 中这个问题没有很好的解决方案......我的解决方案很丑陋,但遗憾的是,这项工作做得很完美:

public static void setMenuSelectedIndex(final JPopupMenu popupMenu, final int index) {
    SwingUtilities.invokeLater(new Runnable(){public void run()
    {
        for (int i=0; i < index+1; i++) {
            popupMenu.dispatchEvent(new KeyEvent(popupMenu, KeyEvent.KEY_PRESSED, 0, 0, KeyEvent.VK_DOWN, '\0'));
        }
    }});
}

如您所见,我基本上是在弹出菜单上模拟“向下”键盘按键...

更好的解决方案可能不是硬编码模拟 VK_DOWN,而是读取 Popup 的输入映射并确定哪个 KeyCode 表示“选择下一个菜单项” - 但我认为我们大多数人都会与这个 hack 相处......

您可能还想查看此方法,该方法在选择菜单项后选择它它利用先前的方法

public static void setSelectedIndexWhenVisible(final JMenu menu, final int index) {
    menu.getPopupMenu().addPopupMenuListener(new PopupMenuListener() {
        @Override
        public void popupMenuWillBecomeVisible(PopupMenuEvent e) {
            PopupUtils.setMenuSelectedIndex(menu.getPopupMenu(), index);
            menu.getPopupMenu().removePopupMenuListener(this);
        }

        @Override
        public void popupMenuWillBecomeInvisible(PopupMenuEvent e) {
        }

        @Override
        public void popupMenuCanceled(PopupMenuEvent e) {
        }
    });
}
于 2011-10-13T13:03:34.090 回答
0

这很奇怪。

我在 Windows 上进行了尝试,在Java 1.5.0_08甚至1.6.0_07中,第一个元素会自动选择,正如您所期望的那样。

所以我用1.6.0_11尝试了它,它不再起作用,最初没有选择第一个元素。在 selectionModel 中选择元素似乎没有帮助。

一种解决方法(我一点也不自豪)是在显示弹出菜单后使用MouseEvent的坐标自动移动鼠标。也许有人有更好的主意?

import java.awt.AWTException;
import java.awt.Robot;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;

import javax.swing.JFrame;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;

public class SelectedPopupMenu extends JFrame {

    public SelectedPopupMenu() {
        addMouseListener(new MouseAdapter() {
            public void mouseClicked(final MouseEvent e) {
                JPopupMenu popupMenu = new JPopupMenu();
                popupMenu.add(new JMenuItem("Test-Item"));
                popupMenu.add(new JMenuItem("Test-Item-2"));
                // do not care to really hit the center of the popup
                popupMenu.show(SelectedPopupMenu.this, e.getX() - 30, e.getY() - 10);
                try {
                    // shake mouse, so that first element is selected even in Java 1.6.0_11
                    Robot robot = new Robot();
                    robot.mouseMove(e.getX() + 1, e.getY());
                    robot.mouseMove(e.getX(), e.getY());
                } catch (AWTException ex) {
                    ex.printStackTrace();
                }
            }
        });
    }

    public static void main(String[] args) {
        JFrame frame = new SelectedPopupMenu();
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.setSize(800, 600);
        frame.setVisible(true);
    }
}
于 2009-01-28T19:10:37.037 回答