1

我正在尝试开发一个由系统托盘图标提供的 Mac OsX 应用程序,因此在第一次尝试使用最简单的代码来实现它之后,我注意到 mac osX(10.8)上的每个应用程序托盘图标(系统和用户应用程序)都允许通过左键和右键单击来激活相关弹出菜单,但在我的项目中,只有左(MouseEvent.BOTTON1)按钮会导致弹出菜单下拉。这是我的代码:

public class SystemTrayDemo
{
    private SystemTray tray;
    private TrayIcon tray_icon;

    public SystemTrayDemo()
    {
        if (!SystemTray.isSupported())
        {
            JOptionPane.showMessageDialog(null, "System tray not supported!");
            return;
        }
        else
           tray = SystemTray.getSystemTray();

        final PopupMenu popup = new PopupMenu(); 

        MenuItem exit = new MenuItem("Exit");

        exit.addActionListener(new ActionListener()
        {
            public void actionPerformed(ActionEvent e)
            {
                if (tray != null)
                {                    
                    tray.remove(tray_icon);
                    System.exit(0);
                }
            }
        });

        popup.add(exit);

        //add tray icon
        tray_icon = new TrayIcon(getIcon("images/wifi.png"), "Open documents...", popup);
        tray_icon.setImageAutoSize(true);

        try
        {            
            tray.add(tray_icon); // adds icon
        }
        catch (AWTException ex) {}
    }


    private Image getIcon(String name)
    {
        URL _url = getClass().getResource(name);
        return new ImageIcon(_url).getImage();
    }

    public static void main(String args[])
    {
        new SystemTrayDemo();
    }
}

但是我已经说过了,只能通过鼠标左键单击。因此,在进一步的尝试中,我尝试使用 MouseListener 模仿每个其他应用程序的托盘图标的行为,并使用 dispatchEvent() 方法在右键单击事件上触发左键事件,如下所示:

    public static void fireMouseEvent(Component c)
    {
        MouseEvent me = new MouseEvent(c, // which
                    MouseEvent.MOUSE_CLICKED, // what
                    System.currentTimeMillis(), // when
                    MouseEvent.BUTTON1_MASK, // no modifiers
                    0, 0, // where: at (10, 10}
                    1, // only 1 click 
                    true); // popup trigger

        c.dispatchEvent(me);
    }

该事件将由鼠标侦听器处理,但显然 TrayIcon 类不是 Component 子类,因此 MouseEvent 的源为空,我得到一个 NPE。这是我的鼠标监听器:

    class MouseAdapt extends java.awt.event.MouseAdapter
    {

        public void mouseClicked(java.awt.event.MouseEvent me)
        {
            int button = me.getButton();

            if(button == java.awt.event.MouseEvent.BUTTON3)
            {
                fireMouseEvent(me.getComponent());
            }
        }
    }

    try
    {            
        tray.add(tray_icon); // aggiungi l'icona
        tray_icon.addMouseListener(new MouseAdapt());
    }
    catch (AWTException ex) {}

对不起我的英语,我希望曾经有过这类项目经验的人可以帮助我。我已经搜索了几个小时,但没有运气。感谢您的帮助。

4

1 回答 1

2

编辑:现在有一个库可以在这里解决所有这些问题: https ://github.com/dorkbox/SystemTray


通过左右单击激活 [TrayIcon] 相关弹出菜单

目前这在 Mac + Java 上是不可能的。使用反射来调用底层触发器似乎没有帮助。这是一个错误。

https://bugs.openjdk.java.net/browse/JDK-8041890

只有左侧 (MouseEvent.BOTTON1) 按钮会导致弹出菜单下拉。这是我的代码

即使这在某些 Java 版本(7u79)中也被破坏了,通过升级修复......

https://bugs.openjdk.java.net/browse/JDK-7158615

跨平台 TrayIcon 支持:

尽管有点偏离主题,但我想补充一点,一些项目使用 aJXTrayIcon来完成 Linux/Windows 等中的一些精美下拉菜单。这些也会在 Mac 上造成问题,尽管它现在已经遭受了点击错误以及错误Gnome3 需要完全独立的 hack。但是在 Mac 上,任何使用装饰菜单的尝试都会导致菜单滞留,这对最终用户来说是一种非常糟糕的体验。我决定的解决方案是在Mac 上使用 AWT,在其他所有东西上使用 Swing。Java TrayIcon 支持迫切需要重写。JavaFX 声称有助于这项计划,但它是为 Java 9 准备的。同时,我坚持使用OS-dependent hacks

其他平台的相关托盘问题

此外,一些 Linux 发行版(如 Ubuntu)默认删除了 Unity 桌面中的托盘图标,这引起了更多的麻烦。https://askubuntu.com/a/457212/412004

此外,在 Gtk/Gnome 或 Qt/KDE 驱动的桌面上,图标的透明度被灰色替换(OpenJDK 和 Oracle JRE 都受此影响) https://stackoverflow.com/a/3882028/3196753 http:// bugs.java.com/bugdatabase/view_bug.do?bug_id=6453521

此外,Gnome3 驱动的桌面可能会在错误的角落显示它,根本不显示,或者它可能显示但无法点击(OpenJDK 和 Oracle JRE 都受此影响) https://bugs.debian.org/cgi-bin/bugreport。 cgi?bug=660157 https://bugzilla.redhat.com/show_bug.cgi?id=1014448

除此之外,Windows 上的高 DPI 屏幕还有一个错误绘制图标的错误:Windows 8 Distorts my TrayIcon

所以综上所述,Java中System Tray的状态还可以,但是由于多种因素的结合,在JDK6、JDK7、JDK8中还是比较碎片化和BUG的。

于 2016-03-10T14:50:22.860 回答