将 JMenuItem 添加到已经可见的 JMenu

Add JMenuItem to Already Visible JMenu

我已经在听 MenuListener.menuSelected() to know when a JMenu is showing on the screen. Initially, the JMenu has 1 JMenuItem 说 "Loading..."。经过缓慢的操作,我将JMenuItems添加到JMenuJMenu 继续显示 "Loading..."。如果我 select 另一个 JMenu 并返回,则 JMenu 会显示添加的 JMenuItem。如何使添加的 JMenuItem 立即显示?

这是重现正在发生的事情的代码。

public class AddMenuItem extends JFrame
{
   private final JMenu m_menu = new JMenu("Edit");

   public static void main(String args[])
   {
      new AddMenuItem().
         setVisible(true);
   }

   public AddMenuItem()
   {
      JMenuBar bar;

      bar = new JMenuBar();

      bar.add(m_menu);
      setJMenuBar(bar);
      setSize(600, 400);

      m_menu.addMenuListener(new MenuListener()
      {
         @Override
         public void menuSelected(MenuEvent e)
         {
            JMenuItem loading;

            loading = new JMenuItem("Loading...");

            loading.setEnabled(false);
            m_menu.removeAll();
            m_menu.add(loading);

            // This represents a long running task which updates the menu afterwards
            new Timer(5 * 1000, event -> updateMenu()).start();
         }

         @Override
         public void menuDeselected(MenuEvent e)
         {
            // nothing to do
         }

         @Override
         public void menuCanceled(MenuEvent e)
         {
            // nothing to do
         }
      });
   }

   private void updateMenu()
   {
      m_menu.removeAll();
      m_menu.add(new JMenuItem("1"));
      m_menu.add(new JMenuItem("2"));
      m_menu.add(new JMenuItem("3"));
      m_menu.revalidate();
   }
}

看起来这个 code example 完全符合我的要求,但我不确定他们正在做什么来使屏幕上的可见菜单重绘。

Initially, the JMenu has 1 JMenuItem which says "Loading...".

从侦听器调用的代码在 Event Dispatch Thread (EDT) 上执行。当您执行长 运行 任务时,它会阻止 EDT 响应事件和重新绘制 GUI。

因此,要解决由该菜单项启动的任务需要在单独的线程中执行的问题。您可能要考虑使用 SwingWorker。阅读有关 Concurrency 的 Swing 教程部分,了解有关 EDTSwingWorker 的更多信息。

编辑:

The problem is that once I know what to add to the menu, I don't know how to get the menu to show the added JMenuItems

您从 MenuListenerMenuEvent 得到了 JMenu:

JMenu menu = (JMenu)e.getSource();
menu.add( new JMenuItem( "Loading..." ) );

编辑:

基于 SSCCE,您可以执行以下操作:

   private void updateMenu()
   {
      m_menu.removeAll();
      m_menu.add(new JMenuItem("1"));
      m_menu.add(new JMenuItem("2"));
      m_menu.add(new JMenuItem("3"));
      JPopupMenu popup = m_menu.getPopupMenu();
      popup.pack();
   }

但请记住,因为您的长 运行 任务在单独的线程中执行,您需要使用 SwingUtiltities.invokeLater 调用上述代码,以便代码在 EDT 上执行。

It seems that this code example does exactly what I want but I am not sure what they are doing to make the visible menu repaint on the screen.

我认为他们只是隐藏并再次显示菜单。

不要执行 m_menu.revalidate(),而是执行以下操作...

if (m_menu.isPopupMenuVisible())
{
   m_menu.setPopupMenuVisible(false);
   m_menu.setPopupMenuVisible(true);
}