Как запретить всплывающему окну TrayIcon занимать весь поток диспетчера

У меня есть 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, и оно показывает мне, что поток диспетчера событий очень занят, пока всплывающее окно видно.

анализ jvisualvm

Я мог понять, что взгляд на всплывающее окно становится видимым с помощью метода sun.awt.windows.WTrayIconPeer.showPopupMenu(int, int).

Этот метод отправляет исполняемый объект с помощью EventQueue.invokeLater в поток диспетчеризации событий. В этом runnable вызывается метод sun.awt.windows.WPopupMenuPeer._show. Это метод native, и кажется, что он реализует занятый цикл ожидания, так что поток отправки событий полностью занят этим методом. Таким образом, другие задачи, связанные с пользовательским интерфейсом, не выполняются, пока отображается всплывающее меню.

Кто-нибудь знает обходной путь, который поддерживает реакцию потока отправки событий, пока отображается всплывающее окно TrayIcon?


person René Link    schedule 05.07.2016    source источник


Ответы (1)


Теперь я использую JPopupMenu в качестве обходного пути, но вы не можете добавить JPopupMenu непосредственно к TrayIcon.

Хитрость заключается в том, чтобы написать MouseListener

public class JPopupMenuMouseAdapter extends MouseAdapter {

  private JPopupMenu popupMenu;

  public JPopupMenuMouseAdapter(JPopupMenu popupMenu) {
    this.popupMenu = popupMenu;
  }

  @Override
  public void mouseReleased(MouseEvent e) {
    maybeShowPopup(e);
  }

  @Override
  public void mousePressed(MouseEvent e) {
    maybeShowPopup(e);
  }

  private void maybeShowPopup(MouseEvent e) {
    if (e.isPopupTrigger()) {
      Dimension size = popupMenu.getPreferredSize();
      popupMenu.setLocation(e.getX() - size.width, e.getY() - size.height);
      popupMenu.setInvoker(popupMenu);
      popupMenu.setVisible(true);
    }
  }

и подключите его к TrayIcon

TrayIcon trayIcon = ...;
JPopupMenu popupMenu = ...;

JPopupMenuMouseAdapter jPopupMenuMouseAdapter = new JPopupMenuMouseAdapter(popupMenu);
trayIcon.addMouseListener(jPopupMenuMouseAdapter);   
person René Link    schedule 05.07.2016