使用JNA访问COM接口

时间:2017-05-17 05:29:39

标签: java com jna

我正在尝试使用JNA访问IDesktopWallpaper界面,但我遇到了障碍。

我浏览了ShOljIdl_core.idl(来自Windows 10 SDK)并发现了界面的GUID,如下所示

// IDesktopWallpaper
[
    uuid(B92B56A9-8B55-4E14-9A89-0199BBB6F93B),
    object
]

和实现接口的具体类的GUID

// CLSID_DesktopWallpaper
[uuid(C2CF3110-460E-4fc1-B9D0-8A1C0C9CC4BD)] coclass DesktopWallpaper { interface IDesktopWallpaper; }

所以我按照JDA github中的官方示例编写了以下内容

@ComObject(clsId="{C2CF3110-460E-4fc1-B9D0-8A1C0C9CC4BD}")
public interface DesktopWallpaper extends IUnknown{


}

并在Main

Ole32.INSTANCE.CoInitializeEx(Pointer.NULL, Ole32.COINIT_MULTITHREADED);
try {
    Factory factory = new Factory();
    try {
        DesktopWallpaper dw = factory.createObject(DesktopWallpaper.class);
    } finally {
        factory.disposeAll();
        factory.getComThread().terminate(1 * 1000);
    }

} finally {
    Ole32.INSTANCE.CoUninitialize();
}

但是factory.createObject(DesktopWallpaper.class)抛出No such interface supported(HRESULT: 80004002) (puArgErr=)并且我不知道如何解决这个问题或为什么会发生这种情况。

任何专家都可以告诉我发生了什么吗? (我是一个完整的菜鸟)我会提供任何必要的进一步信息。 JNA可以实现我想要的或者我必须使用像Com4j这样的东西吗?

1 个答案:

答案 0 :(得分:3)

TL;DR

经过大量的谷歌搜索后,我终于开始工作了。问题(至少我目前的理解)是当前的JNA助手只能使用从IDispatch继承的接口。因此,如果有问题的接口(例如IDesktopWallpaper)不从IDispatch继承,那么应该使用vtable进行函数调用。我从this惊人的谷歌论坛帖子中获得了这些信息,其中海报还提供了一个让我入门的代码示例。

以下是SetWallpaper()函数的一些工作代码:

public class DesktopWallpaperHandler extends Unknown{
    private static final GUID CLSID_DesktopWallpaper = new GUID("{C2CF3110-460E-4fc1-B9D0-8A1C0C9CC4BD}");
    private static final GUID IID_IDesktopWallpaper = new GUID("{B92B56A9-8B55-4E14-9A89-0199BBB6F93B}");

    private DesktopWallpaperHandler(Pointer pvInstance) {
        super(pvInstance);
    }

    public static DesktopWallpaperHandler create(){
        PointerByReference p = new PointerByReference();

        WinNT.HRESULT hr = Ole32.INSTANCE.CoCreateInstance(CLSID_DesktopWallpaper, null, WTypes.CLSCTX_SERVER, IID_IDesktopWallpaper, p);
        COMUtils.checkRC(hr);

        DesktopWallpaperHandler handler = new DesktopWallpaperHandler(p.getValue());

        return handler;
    }

    public void SetWallpaper(WTypes.LPWSTR monitor, WTypes.LPWSTR wallpaper){
        int result = this._invokeNativeInt(3, new Object[]{this.getPointer(), monitor, wallpaper});
        COMUtils.checkRC(new HRESULT(result));
    }
}

然后在Main

Ole32.INSTANCE.CoInitializeEx(Pointer.NULL, Ole32.COINIT_MULTITHREADED);
try {
    WTypes.LPWSTR path = new LPWSTR("C:\\Users\\Harry\\Desktop\\1.jpg");
    DesktopWallpaperHandler handler = DesktopWallpaperHandler.create();
    handler.SetWallpaper(null, path);
} finally {
    Ole32.INSTANCE.CoUninitialize();
}

使用IDesktopWallpaper的原始动机是访问淡入过渡效果,现在可以通过添加以下内容来完成:

User32.INSTANCE.SendMessageTimeout(User32.INSTANCE.FindWindow("Progman", null), 0x52c, 0, 0, 0, 500, null);