`

JNA实现窗口最小化

阅读更多

原始发表时间:2009-06-27

 

本文学习过程如下图示(下图为整个工作思维导图的一部分):



开发环境:
JDK 1.5 (编译的目标版本为1.4)
JNA 3.0.9(开发时的版本,将会切换为3.1.0)


    做了几个月的Swing,恼人的问题一个接一个的出现了。
    因为项目需要,得重新定制界面风格,继承可怕的UI接口,去实现ButtonUI、MenuUI等等等等,图片显示、组件定位都得自己来。经搜索,发现网上也有现成的工具类,乍一看感觉真好啊,突然觉得春光明媚,希望之星冉冉升起,心说“Swing社区也很有活力嘛”,定睛一看,才知道都是收费的……随便一个项目的工具包都是几百美金……好了不扯这茬,总之是没戏了……自己写吧……(这里犯了一个错误,其实SUN的官方有一个项目叫做SwingX,在swing lab管理下,里面有不少扩展过的控件,应该说功能已经十分强大,而且参考其源代码也能受益匪浅,结果overlook之后,偶开始硬着头皮写自己的扩展……)

    本来不需要用JNA,也不需要Win32API(小用一下并不复杂,看到此文的读者如果还不会,可以大胆尝试使用,控制windows的窗口很有趣……),结果因为用户的一个很合理的需求,不得不用了。

    定制的界面中,标题栏已经被干掉,最小化、最大化、关闭按钮都替换成了自己写的图形化按钮(如下图)
    因为界面去掉了系统默认的外框,而用自己改造过的图形化标题栏,所以这些按钮的最小化、最大化功能必须自己编写。
    之前最小化没有能够实现,因为为了实现标题栏自定义,应用中的导航界面设置了 setUndecorated(true),即去除了操作系统对于窗口的标题栏特性的支持,所以最小化功能无法使用JFrame中的 void setExtendedState(int state) 来实现。

    这时很自然地会想到Win32API来获取窗口句柄,设置窗口显示状态。
    于是想起了之前使用JNA项目(也是使用JNA才实现了上图中右上角的圆角特效),仔细查看官方文档(一开始没有看完官方文档,导致浪费了2个多小时在捣腾 FindWindow 和 GetWindowText 函数),有段参考代码如下:

user32.EnumWindows(new WNDENUMPROC() {
    int count;
    public boolean callback(Pointer hWnd, Pointer userData) {
        System.out.println("Found window " + hWnd + ", total " + ++count);
        return true;
    }
}, null);

    这段代码的启示意义很重要,虽然这段代码在JNA 3.0.9和3.1.0 已经改变了 callback 方法的参数列表,还是让我写出了以下代码:

        final User32 user32 = User32.INSTANCE;

        user32.EnumWindows(new WNDENUMPROC() {
            int count;

            public boolean callback(HWND wnd, Pointer data) {
                System.out.println("Found window " + wnd + ", total " + ++count);
                int buflen = 150;
                byte[] lpString = new byte[300];
                user32.GetWindowText(wnd, lpString, buflen);
                System.out.println("lpString: " + Native.toString(lpString));
                return true;
            }
        }, null);

    虽然上面的代码没有正确完成打印所有窗口标题的意图(打印出乱码),但是它确实列举了系统当前的各个窗口句柄的信息。其中黑体字太重要了,一开始依葫芦画瓢,竟然错写成return false,结果每次都只能打印一个窗口的句柄信息,好生苦恼……

    通过网上搜索 FindWindow 的用法,其函数原型:
     HWND FindWindow
     (
         LPCTSTR lpClassName,
         LPCTSTR lpWindowName
     );

    理解之后,写出如下代码,用于搜索客户端程序的窗口句柄

    static HWND getUnitFrameWnd(){
        HWND hwnd = User32.INSTANCE.FindWindow(null, "XXX系统功能导航(客户端)");
        if (unitFrameWnd == null) {
            unitFrameWnd = hwnd;
        }
        return unitFrameWnd;
    }

    应用程序的导航界面 是整个应用所有窗口的父窗口,所以它的title (这里标题取值为“XXX系统功能导航(客户端)”)就可以作为 FindWindow 的参数 lpWindowName ,至于Java的JFrame程序对应于参数 lpClassName 的取值为“SunAwtFrame ”(猜猜我是怎么知道的,用了VS 2008里附带的Spy++,专门用于查询窗口句柄的信息,装一下VC++好像就会附带了,不一定要VS这个庞然大物……)

----------------------------
注:这边有个教训,一开始以为参数为空,就觉得传内容为空的字符串也可以,结果发现返回句柄信息总是为空,比如
User32.INSTANCE.FindWindow("", "XXX系统功能导航(客户端)");
就无法找到我们需要的窗口句柄,如果我们无法给出某个参数的值,就必须将该参数设置为null(这里参数 lpClassName 应该 置为null)
----------------------------

    后来查询到Win32API中,控制窗口最小化、最大化比较好的函数是 ShowWindow ,因为它在修改窗口状态的同时,立即应用效果。但是遍寻JNA 3.0.9里的 User32 接口也没有找到 ShowWindow 的影子,怎么办呢?
    查询MSDN文档,发现 ShowWindow 函数说明如下:
BOOL ShowWindow(HWND hWnd,int nCmdShow)

    另外关于参数 nCmdShow ,找到几个可用的常量值分别如下:
int SW_MAXIMIZE = 0x03;
int SW_MINIMIZE = 0x06;
int SW_RESTORE = 0x09;

    看字面意思就知道是我们正在寻求的东东。好吧,学习 JNA 源代码User32 来写一个接口 Win32API ,将本地方法映射到Java的源代码,可以写出下面的代码:

public interface Win32API extends User32 {
    /**
     * 最大化窗口
     */
    public static int SW_MAXIMIZE = 0x03;
    /**
     * 最小化窗口
     */
    public static int SW_MINIMIZE = 0x06;
    /**
     * 恢复窗口
     */
    public static int SW_RESTORE = 0x09;
    Win32API INSTANCE = (Win32API) Native.loadLibrary("user32", Win32API.class, DEFAULT_OPTIONS);

    /**
     * Updated at 下午03:44:53, on 2009-6-26<br>
     * 获取桌面句柄
     *
     * @return
     * @author Caesar
     */
    public HWND GetDesktopWindow();

    /**
     * Updated at 下午03:44:59, on 2009-6-26<br>
     * 设置窗口句柄的显示状态
     *
     * @param wnd
     * @param nCmdShow
     *            可选值为SW_MINIMIZE - 最小化, SW_MAXIMIZE - 最大化
     * @return
     * @author Caesar
     * @see #SW_MINIMIZE
     * @see #SW_MAXIMIZE
     */
    public boolean ShowWindow(HWND wnd, int nCmdShow);
}

    我们需要的工具已经收入囊中,接下来就是在最小化按钮的事件处理上做点工作就OK了。

        WindowButton btnWinMin = new WindowButton(winMinIcon, parent) {
            private static final long serialVersionUID = 1L;
            protected void actionPerformed(Container frame) {
                HWND mainFrame = getUnitFrameWnd ();  // 这个函数的代码在前面已有提供,获取客户端主界面的窗口句柄
                Win32API.INSTANCE.ShowWindow(mainFrame, Win32API.SW_MINIMIZE);
            }
        };

    类 WindowButton 继承自JButton,通过提供的图标对象 winMinIcon 来显示定制的最小化按钮效果,其代码如下:

public class WindowButton extends RolloverBrighterButton implements ActionListener {
    private static final long serialVersionUID = 1L;
    private Container parentComponent;

    public WindowButton() {
        super();
    }

    public WindowButton(Icon icon, Container parentComponent) {
        super(icon);
        this.parentComponent = parentComponent;
        addActionListener(this);
    }

    public void actionPerformed(ActionEvent e) {
        actionPerformed(parentComponent);
    }

    protected void actionPerformed(Container frame) {

    }
}

    至此,通过Win32API的调用,实现了定制的最小化按钮的功能。

 

 

 

后记:

 

朋友问:

..难道重写标题必须用jna吗?
http://www.blogjava.net/javagui/archive/2007/11/03/157948.html
可以看下这篇文章..

我回答:
抱歉,让你困扰了,这边我的原因没有写明
因为我使用的L&F是windows look & feel,项目没有给我大量的时间重新开发一套L&F,另外我当时没有来得及仔细的学习OpenSwing,所以在界面外观上想了点偷懒的办法,就是主界面使用自制的风格,按美工给的图片来确定整体风格,一些小的对话窗口、录入框之类的就直接沿用windows L&F
但是使用Windows L&F有个麻烦,就是翻阅源码的时候,发现L&F中各个类基本上是在进行底层的调用(可能直接调用某些C函数库,现在也没有去深究,只是猜测),一看头大了,就想了个歪招,把JFrame和JDialog的外框给去掉了,套上自己画的,然后通过JNA调用Win32实现了窗口最大化、最小化等功能……
原因如上。 :-)
分享到:
评论

相关推荐

    Java实现获取窗口句柄并操作窗口jna-4.4.0

    通过窗口句柄,我们可以执行诸如移动、最大化、最小化、关闭等窗口操作。在Java中,直接操作窗口句柄通常是不可能的,但借助JNA,我们可以调用Windows API来实现这些功能。 首先,我们需要引入JNA库,可以在项目中...

    取焦点窗口句柄2.rar

    在Windows编程中,获取焦点窗口句柄通常用于实现某些自动化任务、监控用户行为或者控制窗口交互。以下是一些相关的知识点: 1. **GetForegroundWindow 函数**:这是Windows API中最常用的获取焦点窗口句柄的函数。...

    给标题栏添加额外的按钮(功能)

    2. **扩展按钮**: 常见的扩展按钮包括“最小化”、“最大化/还原”和“关闭”之外的其他功能,如“帮助”、“设置”或“打印”。这些按钮可以增加用户的便利性,例如点击“设置”按钮可以直接打开应用程序的配置界面...

    java 远程桌面功能实现

    该库基于JNA(Java Native Access)技术,能够调用操作系统级别的RDP接口,实现与Windows、Linux等系统之间的远程连接。 3. **JNA(Java Native Access)**: JNA是Java平台上的一个库,允许Java代码直接调用操作...

    JAVA托盘程序____systray4j

    这个开源工具使得Java应用能够更好地融入用户的桌面环境,即使窗口被最小化,用户也能通过托盘图标与应用进行交互。 在源代码中,我们可能会看到以下几个关键知识点: 1. **JNA(Java Native Access)**: ...

    lanslim聊天工具

    2. "tray.dll":这通常是一个动态链接库文件,用于实现系统托盘图标和相关功能,如最小化到系统托盘。 3. "lanslim.exe":这是LANSLIM主程序的可执行文件,负责运行和管理整个应用程序。 4. "jna.jar":Java Native ...

    毕业答辩-JAVASQL电子通讯录带系统托盘(论文源代码).rar

    在本项目中,系统托盘功能允许用户将通讯录程序最小化到系统托盘,节省桌面空间,提高用户体验。这通常通过JNI(Java Native Interface)或者特定的库如JNA(Java Native Access)实现,调用操作系统级别的API来完成...

    操作其它程序中的超级列表框.e.rar

    Python、Java等高级语言也可以通过相应库(如Python的pywin32,Java的JNA或JInterop)来实现。 8. **错误处理和兼容性**: 在实际应用中,考虑到软件版本差异、控件的动态加载、用户权限变化等因素,错误处理和...

Global site tag (gtag.js) - Google Analytics