У меня есть Java-приложение, которое использует JFrame
, а также TrayIcon
, и я добавил PopupMenu
к TrayIcon
.
Когда я нажимаю на TrayIcon
, появляется всплывающее меню, но основной кадр зависает, пока виден PopupMenu
.
Моей первой мыслью было, что поток отправки событий кем-то занят. Поэтому я написал небольшой пример приложения, в котором используется Swing Worker и индикатор выполнения.
public class TrayIconTest {
public static void main(String[] args) throws AWTException {
JFrame frame = new JFrame("TrayIconTest");
frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
Container contentPane = frame.getContentPane();
JProgressBar jProgressBar = new JProgressBar();
BoundedRangeModel boundedRangeModel = jProgressBar.getModel();
contentPane.add(jProgressBar);
boundedRangeModel.setMinimum(0);
boundedRangeModel.setMaximum(100);
PopupMenu popup = new PopupMenu();
TrayIcon trayIcon =
new TrayIcon(new BufferedImage(64, 64, BufferedImage.TYPE_INT_RGB), "TEST");
// Create a popup menu components
MenuItem aboutItem = new MenuItem("About");
popup.add(aboutItem);
trayIcon.setPopupMenu(popup);
SystemTray tray = SystemTray.getSystemTray();
tray.add(trayIcon);
IndeterminateSwingWorker indeterminateSwingWorker = new IndeterminateSwingWorker(boundedRangeModel);
indeterminateSwingWorker.execute();
frame.setSize(640, 80);
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
}
class IndeterminateSwingWorker extends SwingWorker<Void, Integer> {
private BoundedRangeModel boundedRangeModel;
public IndeterminateSwingWorker(BoundedRangeModel boundedRangeModel) {
this.boundedRangeModel = boundedRangeModel;
}
@Override
protected Void doInBackground() throws Exception {
int i = 0;
long start = System.currentTimeMillis();
long runtime = TimeUnit.MILLISECONDS.convert(5, TimeUnit.MINUTES);
while (true) {
i++;
i %= boundedRangeModel.getMaximum();
publish(i);
Thread.sleep(20);
long now = System.currentTimeMillis();
if ((now - start) > runtime) {
break;
}
System.out.println("i = " + i);
}
return null;
}
@Override
protected void process(List<Integer> chunks) {
for (Integer chunk : chunks) {
boundedRangeModel.setValue(chunk);
}
}
}
Как воспроизвести
Когда вы запустите пример приложения, вы увидите рамку с индикатором выполнения. SwingWorker
имитирует действие прогресса.
SwingWorker
публикует значение прогресса, а также распечатывает его в System.out
.
Когда я нажимаю на значок в трее («черный»), индикатор выполнения зависает и появляется всплывающее меню. Я вижу, что SwingWorker все еще работает в фоновом режиме, потому что он выводит значение прогресса в командную строку.
Расследование
Итак, я взглянул на приложение, использующее jvisualvm
, и оно показывает мне, что поток диспетчера событий очень занят, пока всплывающее окно видно.
Я мог понять, что взгляд на всплывающее окно становится видимым с помощью метода sun.awt.windows.WTrayIconPeer.showPopupMenu(int, int)
.
Этот метод отправляет исполняемый объект с помощью EventQueue.invokeLater
в поток диспетчеризации событий. В этом runnable вызывается метод sun.awt.windows.WPopupMenuPeer._show
. Это метод native
, и кажется, что он реализует занятый цикл ожидания, так что поток отправки событий полностью занят этим методом. Таким образом, другие задачи, связанные с пользовательским интерфейсом, не выполняются, пока отображается всплывающее меню.
Кто-нибудь знает обходной путь, который поддерживает реакцию потока отправки событий, пока отображается всплывающее окно TrayIcon
?