将 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..."。经过缓慢的操作,我将JMenuItem
s添加到JMenu
。 JMenu
继续显示 "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 教程部分,了解有关 EDT
和 SwingWorker
的更多信息。
编辑:
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
您从 MenuListener
的 MenuEvent
得到了 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);
}
我已经在听 MenuListener.menuSelected() to know when a JMenu is showing on the screen. Initially, the JMenu
has 1 JMenuItem 说 "Loading..."。经过缓慢的操作,我将JMenuItem
s添加到JMenu
。 JMenu
继续显示 "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 教程部分,了解有关 EDT
和 SwingWorker
的更多信息。
编辑:
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
您从 MenuListener
的 MenuEvent
得到了 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);
}