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实现了窗口最大化、最小化等功能……
原因如上。 :-)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JNA(Java Native Access)是一个开源的Java库,它提供了一种简单的方式来访问本地操作系统的API。使用JNA,我们可以轻松地调用Windows API函数来操作窗口。以下是一些常用的JNA窗口操作API: 1. User32.dll User32.dll是Windows API中用于用户界面操作的动态链接库,我们可以使用JNA来调用其中的函数。以下是一些常用的User32.dll函数: - FindWindow:查找窗口句柄。 - SetForegroundWindow:将指定窗口置于最前面。 - GetWindowText:获取窗口标题。 - ShowWindow:显示或隐藏指定窗口。 - SendMessage:向指定窗口发送消息。 以下是一个示例代码,演示如何使用JNA调用User32.dll中的函数: ``` import com.sun.jna.Native; import com.sun.jna.platform.win32.User32; import com.sun.jna.platform.win32.WinDef; public class Main { public static void main(String[] args) { // 查找窗口句柄 WinDef.HWND hwnd = User32.INSTANCE.FindWindow(null, "窗口标题"); // 将窗口置于最前面 User32.INSTANCE.SetForegroundWindow(hwnd); // 获取窗口标题 byte[] title = new byte[1024]; User32.INSTANCE.GetWindowText(hwnd, title, 1024); System.out.println("窗口标题:" + Native.toString(title)); // 隐藏窗口 User32.INSTANCE.ShowWindow(hwnd, User32.SW_HIDE); // 发送消息给窗口 User32.INSTANCE.SendMessage(hwnd, User32.WM_CLOSE, null, null); } } ``` 2. Kernel32.dll Kernel32.dll是Windows API中用于操作系统核心功能的动态链接库,我们可以使用JNA来调用其中的函数。以下是一些常用的Kernel32.dll函数: - GetCurrentThreadId:获取当前线程ID。 - GetCurrentProcessId:获取当前进程ID。 - GetSystemMetrics:获取系统指定参数的信息。 以下是一个示例代码,演示如何使用JNA调用Kernel32.dll中的函数: ``` import com.sun.jna.Native; import com.sun.jna.platform.win32.Kernel32; public class Main { public static void main(String[] args) { // 获取当前线程ID int threadId = Kernel32.INSTANCE.GetCurrentThreadId(); System.out.println("当前线程ID:" + threadId); // 获取当前进程ID int processId = Kernel32.INSTANCE.GetCurrentProcessId(); System.out.println("当前进程ID:" + processId); // 获取屏幕分辨率 int screenWidth = Kernel32.INSTANCE.GetSystemMetrics(Kernel32.SM_CXSCREEN); int screenHeight = Kernel32.INSTANCE.GetSystemMetrics(Kernel32.SM_CYSCREEN); System.out.println("屏幕分辨率:" + screenWidth + "x" + screenHeight); } } ``` 以上是一些常用的JNA窗口操作API,你可以根据实际需求选择合适的函数来操作窗口
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值