2016-07-05 9 views
1

TrayIconと同様にJFrameを使用するJavaアプリケーションがあり、TrayIconPopupMenuを追加しました。TrayIconポップアップがディスパッチャスレッド全体を占めるのを防ぐ方法

TrayIconをクリックすると、ポップアップメニューが表示されますが、PopupMenuが表示されている限り、メインフレームがフリーズします。

私の最初の考えは、イベントディスパッチスレッドが誰かによって占有されていたことでした。だから 私はスイングワーカーと進行状況バーを使用する小さなサンプルアプリケーションを書いた。あなたがサンプルアプリケーションを起動するときは、プログレスバーと枠が表示されます

を再現する方法

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 analysis

私はポップアップを見ては、法sun.awt.windows.WTrayIconPeer.showPopupMenu(int, int)により可視化されていることを見つけ出すことができます。

このメソッドは、EventQueue.invokeLaterを使用して実行可能なイベントをイベントディスパッチスレッドに送信します。この実行可能ファイルでは、メソッドsun.awt.windows.WPopupMenuPeer._showが呼び出されます。このメソッドはnativeです。このメソッドでは、イベントディスパッチスレッドが完全に占有されるようにビジー待機ループを実装しているようです。 ポップアップメニューが表示されている限り、他のui関連タスクは実行されません。

誰でも、TrayIconポップアップが表示されている間、イベントディスパッチスレッドを応答し続ける回避策を知っていますか?

答えて

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); 
に接続することです
関連する問題